mirror of https://github.com/apache/lucene.git
LUCENE-4061: fixed another potential concurrency issue; deleted TestIndexClose - the test-framework guarantees we close all our resources after us
git-svn-id: https://svn.apache.org/repos/asf/lucene/dev/trunk@1339491 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
parent
bc33ed0ef2
commit
3e7339af68
|
@ -36,7 +36,6 @@ import org.apache.lucene.index.IndexWriter;
|
||||||
import org.apache.lucene.index.IndexWriterConfig;
|
import org.apache.lucene.index.IndexWriterConfig;
|
||||||
import org.apache.lucene.index.IndexWriterConfig.OpenMode;
|
import org.apache.lucene.index.IndexWriterConfig.OpenMode;
|
||||||
import org.apache.lucene.index.LogByteSizeMergePolicy;
|
import org.apache.lucene.index.LogByteSizeMergePolicy;
|
||||||
import org.apache.lucene.index.MultiFields;
|
|
||||||
import org.apache.lucene.index.SegmentInfos;
|
import org.apache.lucene.index.SegmentInfos;
|
||||||
import org.apache.lucene.index.Terms;
|
import org.apache.lucene.index.Terms;
|
||||||
import org.apache.lucene.index.TermsEnum;
|
import org.apache.lucene.index.TermsEnum;
|
||||||
|
@ -46,7 +45,6 @@ import org.apache.lucene.store.Directory;
|
||||||
import org.apache.lucene.store.LockObtainFailedException;
|
import org.apache.lucene.store.LockObtainFailedException;
|
||||||
import org.apache.lucene.store.NativeFSLockFactory;
|
import org.apache.lucene.store.NativeFSLockFactory;
|
||||||
import org.apache.lucene.store.SimpleFSLockFactory;
|
import org.apache.lucene.store.SimpleFSLockFactory;
|
||||||
import org.apache.lucene.util.Bits;
|
|
||||||
import org.apache.lucene.util.BytesRef;
|
import org.apache.lucene.util.BytesRef;
|
||||||
import org.apache.lucene.util.Version;
|
import org.apache.lucene.util.Version;
|
||||||
|
|
||||||
|
@ -232,7 +230,7 @@ public class DirectoryTaxonomyWriter implements TaxonomyWriter {
|
||||||
// Make sure that the taxonomy always contain the root category
|
// Make sure that the taxonomy always contain the root category
|
||||||
// with category id 0.
|
// with category id 0.
|
||||||
addCategory(new CategoryPath());
|
addCategory(new CategoryPath());
|
||||||
refreshReader();
|
refreshInternalReader();
|
||||||
} else {
|
} else {
|
||||||
// There are some categories on the disk, which we have not yet
|
// There are some categories on the disk, which we have not yet
|
||||||
// read into the cache, and therefore the cache is incomplete.
|
// read into the cache, and therefore the cache is incomplete.
|
||||||
|
@ -289,14 +287,14 @@ public class DirectoryTaxonomyWriter implements TaxonomyWriter {
|
||||||
new LogByteSizeMergePolicy());
|
new LogByteSizeMergePolicy());
|
||||||
}
|
}
|
||||||
|
|
||||||
// Currently overridden by a unit test that verifies that every index we open is close()ed.
|
/** Opens a {@link DirectoryReader} from the internal {@link IndexWriter}. */
|
||||||
/**
|
private synchronized void openInternalReader() throws IOException {
|
||||||
* Open an {@link IndexReader} from the internal {@link IndexWriter}, by
|
// verify that the taxo-writer hasn't been closed on us. the method is
|
||||||
* calling {@link IndexReader#open(IndexWriter, boolean)}. Extending classes can override
|
// synchronized since it may be called from a non sync'ed block, and it
|
||||||
* this method to return their own {@link IndexReader}.
|
// needs to protect against close() happening concurrently.
|
||||||
*/
|
ensureOpen();
|
||||||
protected DirectoryReader openReader() throws IOException {
|
assert reader == null : "a reader is already open !";
|
||||||
return DirectoryReader.open(indexWriter, true);
|
reader = DirectoryReader.open(indexWriter, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -398,7 +396,7 @@ public class DirectoryTaxonomyWriter implements TaxonomyWriter {
|
||||||
// We need to get an answer from the on-disk index. If a reader
|
// We need to get an answer from the on-disk index. If a reader
|
||||||
// is not yet open, do it now:
|
// is not yet open, do it now:
|
||||||
if (reader == null) {
|
if (reader == null) {
|
||||||
reader = openReader();
|
openInternalReader();
|
||||||
}
|
}
|
||||||
|
|
||||||
int base = 0;
|
int base = 0;
|
||||||
|
@ -442,7 +440,7 @@ public class DirectoryTaxonomyWriter implements TaxonomyWriter {
|
||||||
return cache.get(categoryPath, prefixLen);
|
return cache.get(categoryPath, prefixLen);
|
||||||
}
|
}
|
||||||
if (reader == null) {
|
if (reader == null) {
|
||||||
reader = openReader();
|
openInternalReader();
|
||||||
}
|
}
|
||||||
|
|
||||||
int base = 0;
|
int base = 0;
|
||||||
|
@ -615,7 +613,7 @@ public class DirectoryTaxonomyWriter implements TaxonomyWriter {
|
||||||
// Because this is a slow operation, cache implementations are
|
// Because this is a slow operation, cache implementations are
|
||||||
// expected not to delete entries one-by-one but rather in bulk
|
// expected not to delete entries one-by-one but rather in bulk
|
||||||
// (LruTaxonomyWriterCache removes the 2/3rd oldest entries).
|
// (LruTaxonomyWriterCache removes the 2/3rd oldest entries).
|
||||||
refreshReader();
|
refreshInternalReader();
|
||||||
cacheIsComplete = false;
|
cacheIsComplete = false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -623,12 +621,12 @@ public class DirectoryTaxonomyWriter implements TaxonomyWriter {
|
||||||
private void addToCache(CategoryPath categoryPath, int prefixLen, int id)
|
private void addToCache(CategoryPath categoryPath, int prefixLen, int id)
|
||||||
throws IOException {
|
throws IOException {
|
||||||
if (cache.put(categoryPath, prefixLen, id)) {
|
if (cache.put(categoryPath, prefixLen, id)) {
|
||||||
refreshReader();
|
refreshInternalReader();
|
||||||
cacheIsComplete = false;
|
cacheIsComplete = false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
protected synchronized void refreshReader() throws IOException {
|
private synchronized void refreshInternalReader() throws IOException {
|
||||||
if (reader != null) {
|
if (reader != null) {
|
||||||
DirectoryReader r2 = DirectoryReader.openIfChanged(reader);
|
DirectoryReader r2 = DirectoryReader.openIfChanged(reader);
|
||||||
if (r2 != null) {
|
if (r2 != null) {
|
||||||
|
@ -648,7 +646,7 @@ public class DirectoryTaxonomyWriter implements TaxonomyWriter {
|
||||||
public synchronized void commit() throws CorruptIndexException, IOException {
|
public synchronized void commit() throws CorruptIndexException, IOException {
|
||||||
ensureOpen();
|
ensureOpen();
|
||||||
indexWriter.commit(combinedCommitData(null));
|
indexWriter.commit(combinedCommitData(null));
|
||||||
refreshReader();
|
refreshInternalReader();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -674,7 +672,7 @@ public class DirectoryTaxonomyWriter implements TaxonomyWriter {
|
||||||
public synchronized void commit(Map<String,String> commitUserData) throws CorruptIndexException, IOException {
|
public synchronized void commit(Map<String,String> commitUserData) throws CorruptIndexException, IOException {
|
||||||
ensureOpen();
|
ensureOpen();
|
||||||
indexWriter.commit(combinedCommitData(commitUserData));
|
indexWriter.commit(combinedCommitData(commitUserData));
|
||||||
refreshReader();
|
refreshInternalReader();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -759,7 +757,7 @@ public class DirectoryTaxonomyWriter implements TaxonomyWriter {
|
||||||
// TODO (Facet): we should probably completely clear the cache before starting
|
// TODO (Facet): we should probably completely clear the cache before starting
|
||||||
// to read it?
|
// to read it?
|
||||||
if (reader == null) {
|
if (reader == null) {
|
||||||
reader = openReader();
|
openInternalReader();
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!cache.hasRoom(reader.numDocs())) {
|
if (!cache.hasRoom(reader.numDocs())) {
|
||||||
|
@ -826,7 +824,7 @@ public class DirectoryTaxonomyWriter implements TaxonomyWriter {
|
||||||
private synchronized ParentArray getParentArray() throws IOException {
|
private synchronized ParentArray getParentArray() throws IOException {
|
||||||
if (parentArray==null) {
|
if (parentArray==null) {
|
||||||
if (reader == null) {
|
if (reader == null) {
|
||||||
reader = openReader();
|
openInternalReader();
|
||||||
}
|
}
|
||||||
parentArray = new ParentArray();
|
parentArray = new ParentArray();
|
||||||
parentArray.refresh(reader);
|
parentArray.refresh(reader);
|
||||||
|
@ -888,18 +886,6 @@ public class DirectoryTaxonomyWriter implements TaxonomyWriter {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Expert: This method is only for expert use.
|
|
||||||
* Note also that any call to refresh() will invalidate the returned reader,
|
|
||||||
* so the caller needs to take care of appropriate locking.
|
|
||||||
*
|
|
||||||
* @return lucene indexReader
|
|
||||||
*/
|
|
||||||
DirectoryReader getInternalIndexReader() {
|
|
||||||
ensureOpen();
|
|
||||||
return this.reader;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Mapping from old ordinal to new ordinals, used when merging indexes
|
* Mapping from old ordinal to new ordinals, used when merging indexes
|
||||||
* wit separate taxonomies.
|
* wit separate taxonomies.
|
||||||
|
|
|
@ -1,193 +0,0 @@
|
||||||
package org.apache.lucene.facet.taxonomy.directory;
|
|
||||||
|
|
||||||
import java.io.IOException;
|
|
||||||
import java.util.Collections;
|
|
||||||
import java.util.HashSet;
|
|
||||||
import java.util.IdentityHashMap;
|
|
||||||
import java.util.Set;
|
|
||||||
|
|
||||||
import org.apache.lucene.index.CorruptIndexException;
|
|
||||||
import org.apache.lucene.index.DirectoryReader;
|
|
||||||
import org.apache.lucene.index.IndexWriter;
|
|
||||||
import org.apache.lucene.index.IndexWriterConfig;
|
|
||||||
import org.apache.lucene.index.IndexWriterConfig.OpenMode;
|
|
||||||
import org.apache.lucene.store.Directory;
|
|
||||||
import org.apache.lucene.store.LockObtainFailedException;
|
|
||||||
import org.junit.Test;
|
|
||||||
|
|
||||||
import org.apache.lucene.util.LuceneTestCase;
|
|
||||||
import org.apache.lucene.analysis.MockAnalyzer;
|
|
||||||
import org.apache.lucene.analysis.MockTokenizer;
|
|
||||||
import org.apache.lucene.facet.taxonomy.CategoryPath;
|
|
||||||
import org.apache.lucene.facet.taxonomy.InconsistentTaxonomyException;
|
|
||||||
import org.apache.lucene.facet.taxonomy.directory.DirectoryTaxonomyReader;
|
|
||||||
import org.apache.lucene.facet.taxonomy.directory.DirectoryTaxonomyWriter;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* 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.
|
|
||||||
*/
|
|
||||||
|
|
||||||
/**
|
|
||||||
* This test case attempts to catch index "leaks" in LuceneTaxonomyReader/Writer,
|
|
||||||
* i.e., cases where an index has been opened, but never closed; In that case,
|
|
||||||
* Java would eventually collect this object and close the index, but leaving
|
|
||||||
* the index open might nevertheless cause problems - e.g., on Windows it prevents
|
|
||||||
* deleting it.
|
|
||||||
*/
|
|
||||||
public class TestIndexClose extends LuceneTestCase {
|
|
||||||
|
|
||||||
@Test
|
|
||||||
public void testLeaks() throws Exception {
|
|
||||||
LeakChecker checker = new LeakChecker();
|
|
||||||
Directory dir = newDirectory();
|
|
||||||
DirectoryTaxonomyWriter tw = checker.openWriter(dir);
|
|
||||||
tw.close();
|
|
||||||
assertEquals(0, checker.nopen());
|
|
||||||
|
|
||||||
tw = checker.openWriter(dir);
|
|
||||||
tw.addCategory(new CategoryPath("animal", "dog"));
|
|
||||||
tw.close();
|
|
||||||
assertEquals(0, checker.nopen());
|
|
||||||
|
|
||||||
DirectoryTaxonomyReader tr = checker.openReader(dir);
|
|
||||||
tr.getPath(1);
|
|
||||||
tr.refresh();
|
|
||||||
tr.close();
|
|
||||||
assertEquals(0, checker.nopen());
|
|
||||||
|
|
||||||
tr = checker.openReader(dir);
|
|
||||||
tw = checker.openWriter(dir);
|
|
||||||
tw.addCategory(new CategoryPath("animal", "cat"));
|
|
||||||
tr.refresh();
|
|
||||||
tw.commit();
|
|
||||||
tw.close();
|
|
||||||
tr.refresh();
|
|
||||||
tr.close();
|
|
||||||
assertEquals(0, checker.nopen());
|
|
||||||
|
|
||||||
tw = checker.openWriter(dir);
|
|
||||||
for (int i=0; i<10000; i++) {
|
|
||||||
tw.addCategory(new CategoryPath("number", Integer.toString(i)));
|
|
||||||
}
|
|
||||||
tw.close();
|
|
||||||
assertEquals(0, checker.nopen());
|
|
||||||
tw = checker.openWriter(dir);
|
|
||||||
for (int i=0; i<10000; i++) {
|
|
||||||
tw.addCategory(new CategoryPath("number", Integer.toString(i*2)));
|
|
||||||
}
|
|
||||||
tw.close();
|
|
||||||
assertEquals(0, checker.nopen());
|
|
||||||
dir.close();
|
|
||||||
}
|
|
||||||
|
|
||||||
private static class LeakChecker {
|
|
||||||
Set<DirectoryReader> readers = Collections.newSetFromMap(new IdentityHashMap<DirectoryReader,Boolean>());
|
|
||||||
|
|
||||||
int iwriter=0;
|
|
||||||
Set<Integer> openWriters = new HashSet<Integer>();
|
|
||||||
|
|
||||||
LeakChecker() { }
|
|
||||||
|
|
||||||
public DirectoryTaxonomyWriter openWriter(Directory dir) throws CorruptIndexException, LockObtainFailedException, IOException {
|
|
||||||
return new InstrumentedTaxonomyWriter(dir);
|
|
||||||
}
|
|
||||||
|
|
||||||
public DirectoryTaxonomyReader openReader(Directory dir) throws CorruptIndexException, LockObtainFailedException, IOException {
|
|
||||||
return new InstrumentedTaxonomyReader(dir);
|
|
||||||
}
|
|
||||||
|
|
||||||
public int nopen() {
|
|
||||||
int ret=0;
|
|
||||||
for (DirectoryReader r: readers) {
|
|
||||||
if (r.getRefCount() > 0) {
|
|
||||||
System.err.println("reader "+r+" still open");
|
|
||||||
ret++;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
for (int i: openWriters) {
|
|
||||||
System.err.println("writer "+i+" still open");
|
|
||||||
ret++;
|
|
||||||
}
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
private class InstrumentedTaxonomyWriter extends DirectoryTaxonomyWriter {
|
|
||||||
public InstrumentedTaxonomyWriter(Directory dir) throws CorruptIndexException, LockObtainFailedException, IOException {
|
|
||||||
super(dir);
|
|
||||||
}
|
|
||||||
@Override
|
|
||||||
protected DirectoryReader openReader() throws IOException {
|
|
||||||
DirectoryReader r = super.openReader();
|
|
||||||
readers.add(r);
|
|
||||||
return r;
|
|
||||||
}
|
|
||||||
@Override
|
|
||||||
protected synchronized void refreshReader() throws IOException {
|
|
||||||
super.refreshReader();
|
|
||||||
final DirectoryReader r = getInternalIndexReader();
|
|
||||||
if (r != null) readers.add(r);
|
|
||||||
}
|
|
||||||
@Override
|
|
||||||
protected IndexWriter openIndexWriter (Directory directory, IndexWriterConfig config) throws IOException {
|
|
||||||
return new InstrumentedIndexWriter(directory, config);
|
|
||||||
}
|
|
||||||
@Override
|
|
||||||
protected IndexWriterConfig createIndexWriterConfig(OpenMode openMode) {
|
|
||||||
return newIndexWriterConfig(TEST_VERSION_CURRENT, new MockAnalyzer(random(), MockTokenizer.KEYWORD, false))
|
|
||||||
.setOpenMode(openMode).setMergePolicy(newLogMergePolicy());
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
private class InstrumentedTaxonomyReader extends DirectoryTaxonomyReader {
|
|
||||||
public InstrumentedTaxonomyReader(Directory dir) throws CorruptIndexException, LockObtainFailedException, IOException {
|
|
||||||
super(dir);
|
|
||||||
}
|
|
||||||
@Override
|
|
||||||
protected DirectoryReader openIndexReader(Directory dir) throws CorruptIndexException, IOException {
|
|
||||||
DirectoryReader r = super.openIndexReader(dir);
|
|
||||||
readers.add(r);
|
|
||||||
return r;
|
|
||||||
}
|
|
||||||
@Override
|
|
||||||
public synchronized boolean refresh() throws IOException, InconsistentTaxonomyException {
|
|
||||||
final boolean ret = super.refresh();
|
|
||||||
readers.add(getInternalIndexReader());
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private class InstrumentedIndexWriter extends IndexWriter {
|
|
||||||
int mynum;
|
|
||||||
public InstrumentedIndexWriter(Directory d, IndexWriterConfig conf) throws CorruptIndexException, LockObtainFailedException, IOException {
|
|
||||||
super(d, conf);
|
|
||||||
mynum = iwriter++;
|
|
||||||
openWriters.add(mynum);
|
|
||||||
// System.err.println("openedw "+mynum);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void close() throws IOException {
|
|
||||||
super.close();
|
|
||||||
if (!openWriters.contains(mynum)) { // probably can't happen...
|
|
||||||
fail("Writer #"+mynum+" was closed twice!");
|
|
||||||
}
|
|
||||||
openWriters.remove(mynum);
|
|
||||||
// System.err.println("closedw "+mynum);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
Loading…
Reference in New Issue