mirror of https://github.com/apache/lucene.git
LUCENE-1329: enable read-only IndexReaders; the default is to get a read-write IndexReader on open, but in 3.0 this will change to read-only
git-svn-id: https://svn.apache.org/repos/asf/lucene/java/trunk@688323 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
parent
59b8dca69a
commit
f0d2a151ca
|
@ -114,6 +114,14 @@ API Changes
|
||||||
hashCode() and equals() in Token, and fixed all core and contrib
|
hashCode() and equals() in Token, and fixed all core and contrib
|
||||||
analyzers to use the re-use APIs. (DM Smith via Mike McCandless)
|
analyzers to use the re-use APIs. (DM Smith via Mike McCandless)
|
||||||
|
|
||||||
|
18. LUCENE-1329: Add optional readOnly boolean when opening an
|
||||||
|
IndexReader. A readOnly reader is not allowed to make changes
|
||||||
|
(deletions, norms) to the index; in exchanged, the isDeleted
|
||||||
|
method, often a bottleneck when searching with many threads, is
|
||||||
|
not synchronized. The default for readOnly is still false, but in
|
||||||
|
3.0 the default will become true. (Jason Rutherglen via Mike
|
||||||
|
McCandless)
|
||||||
|
|
||||||
Bug fixes
|
Bug fixes
|
||||||
|
|
||||||
1. LUCENE-1134: Fixed BooleanQuery.rewrite to only optimize a single
|
1. LUCENE-1134: Fixed BooleanQuery.rewrite to only optimize a single
|
||||||
|
|
|
@ -43,21 +43,24 @@ abstract class DirectoryIndexReader extends IndexReader {
|
||||||
private SegmentInfos segmentInfos;
|
private SegmentInfos segmentInfos;
|
||||||
private Lock writeLock;
|
private Lock writeLock;
|
||||||
private boolean stale;
|
private boolean stale;
|
||||||
private HashSet synced = new HashSet();
|
private final HashSet synced = new HashSet();
|
||||||
|
|
||||||
/** Used by commit() to record pre-commit state in case
|
/** Used by commit() to record pre-commit state in case
|
||||||
* rollback is necessary */
|
* rollback is necessary */
|
||||||
private boolean rollbackHasChanges;
|
private boolean rollbackHasChanges;
|
||||||
private SegmentInfos rollbackSegmentInfos;
|
private SegmentInfos rollbackSegmentInfos;
|
||||||
|
|
||||||
|
protected boolean readOnly;
|
||||||
|
|
||||||
void init(Directory directory, SegmentInfos segmentInfos, boolean closeDirectory)
|
|
||||||
|
void init(Directory directory, SegmentInfos segmentInfos, boolean closeDirectory, boolean readOnly)
|
||||||
throws IOException {
|
throws IOException {
|
||||||
this.directory = directory;
|
this.directory = directory;
|
||||||
this.segmentInfos = segmentInfos;
|
this.segmentInfos = segmentInfos;
|
||||||
this.closeDirectory = closeDirectory;
|
this.closeDirectory = closeDirectory;
|
||||||
|
this.readOnly = readOnly;
|
||||||
|
|
||||||
if (segmentInfos != null) {
|
if (!readOnly && segmentInfos != null) {
|
||||||
// We assume that this segments_N was previously
|
// We assume that this segments_N was previously
|
||||||
// properly sync'd:
|
// properly sync'd:
|
||||||
for(int i=0;i<segmentInfos.size();i++) {
|
for(int i=0;i<segmentInfos.size();i++) {
|
||||||
|
@ -72,16 +75,16 @@ abstract class DirectoryIndexReader extends IndexReader {
|
||||||
protected DirectoryIndexReader() {}
|
protected DirectoryIndexReader() {}
|
||||||
|
|
||||||
DirectoryIndexReader(Directory directory, SegmentInfos segmentInfos,
|
DirectoryIndexReader(Directory directory, SegmentInfos segmentInfos,
|
||||||
boolean closeDirectory) throws IOException {
|
boolean closeDirectory, boolean readOnly) throws IOException {
|
||||||
super();
|
super();
|
||||||
init(directory, segmentInfos, closeDirectory);
|
init(directory, segmentInfos, closeDirectory, readOnly);
|
||||||
}
|
}
|
||||||
|
|
||||||
static DirectoryIndexReader open(final Directory directory, final boolean closeDirectory, final IndexDeletionPolicy deletionPolicy) throws CorruptIndexException, IOException {
|
static DirectoryIndexReader open(final Directory directory, final boolean closeDirectory, final IndexDeletionPolicy deletionPolicy) throws CorruptIndexException, IOException {
|
||||||
return open(directory, closeDirectory, deletionPolicy, null);
|
return open(directory, closeDirectory, deletionPolicy, null, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
static DirectoryIndexReader open(final Directory directory, final boolean closeDirectory, final IndexDeletionPolicy deletionPolicy, final IndexCommit commit) throws CorruptIndexException, IOException {
|
static DirectoryIndexReader open(final Directory directory, final boolean closeDirectory, final IndexDeletionPolicy deletionPolicy, final IndexCommit commit, final boolean readOnly) throws CorruptIndexException, IOException {
|
||||||
|
|
||||||
SegmentInfos.FindSegmentsFile finder = new SegmentInfos.FindSegmentsFile(directory) {
|
SegmentInfos.FindSegmentsFile finder = new SegmentInfos.FindSegmentsFile(directory) {
|
||||||
|
|
||||||
|
@ -93,9 +96,11 @@ abstract class DirectoryIndexReader extends IndexReader {
|
||||||
DirectoryIndexReader reader;
|
DirectoryIndexReader reader;
|
||||||
|
|
||||||
if (infos.size() == 1) { // index is optimized
|
if (infos.size() == 1) { // index is optimized
|
||||||
reader = SegmentReader.get(infos, infos.info(0), closeDirectory);
|
reader = SegmentReader.get(readOnly, infos, infos.info(0), closeDirectory);
|
||||||
|
} else if (readOnly) {
|
||||||
|
reader = new ReadOnlyMultiSegmentReader(directory, infos, closeDirectory);
|
||||||
} else {
|
} else {
|
||||||
reader = new MultiSegmentReader(directory, infos, closeDirectory);
|
reader = new MultiSegmentReader(directory, infos, closeDirectory, false);
|
||||||
}
|
}
|
||||||
reader.setDeletionPolicy(deletionPolicy);
|
reader.setDeletionPolicy(deletionPolicy);
|
||||||
return reader;
|
return reader;
|
||||||
|
@ -131,7 +136,7 @@ abstract class DirectoryIndexReader extends IndexReader {
|
||||||
DirectoryIndexReader newReader = doReopen(infos);
|
DirectoryIndexReader newReader = doReopen(infos);
|
||||||
|
|
||||||
if (DirectoryIndexReader.this != newReader) {
|
if (DirectoryIndexReader.this != newReader) {
|
||||||
newReader.init(directory, infos, closeDirectory);
|
newReader.init(directory, infos, closeDirectory, readOnly);
|
||||||
newReader.deletionPolicy = deletionPolicy;
|
newReader.deletionPolicy = deletionPolicy;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -45,17 +45,33 @@ import java.util.Collection;
|
||||||
opened already, but it cannot be used to delete documents from the index then.
|
opened already, but it cannot be used to delete documents from the index then.
|
||||||
|
|
||||||
<p>
|
<p>
|
||||||
NOTE: for backwards API compatibility, several methods are not listed
|
<b>NOTE</b>: for backwards API compatibility, several methods are not listed
|
||||||
as abstract, but have no useful implementations in this base class and
|
as abstract, but have no useful implementations in this base class and
|
||||||
instead always throw UnsupportedOperationException. Subclasses are
|
instead always throw UnsupportedOperationException. Subclasses are
|
||||||
strongly encouraged to override these methods, but in many cases may not
|
strongly encouraged to override these methods, but in many cases may not
|
||||||
need to.
|
need to.
|
||||||
</p>
|
</p>
|
||||||
|
|
||||||
|
<p>
|
||||||
|
|
||||||
|
<b>NOTE</b>: as of 2.4, it's possible to open a read-only
|
||||||
|
IndexReader using one of the static open methods that
|
||||||
|
accepts the boolean readOnly parameter. Such a reader has
|
||||||
|
better concurrency as it's not necessary to synchronize on
|
||||||
|
the isDeleted method. Currently the default for readOnly
|
||||||
|
is false, meaning if not specified you will get a
|
||||||
|
read/write IndexReader. But in 3.0 this default will
|
||||||
|
change to true, meaning you must explicitly specify false
|
||||||
|
if you want to make changes with the resulting IndexReader.
|
||||||
|
</p>
|
||||||
|
|
||||||
@version $Id$
|
@version $Id$
|
||||||
*/
|
*/
|
||||||
public abstract class IndexReader {
|
public abstract class IndexReader {
|
||||||
|
|
||||||
|
// NOTE: in 3.0 this will change to true
|
||||||
|
final static boolean READ_ONLY_DEFAULT = false;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Constants describing field properties, for example used for
|
* Constants describing field properties, for example used for
|
||||||
* {@link IndexReader#getFieldNames(FieldOption)}.
|
* {@link IndexReader#getFieldNames(FieldOption)}.
|
||||||
|
@ -181,46 +197,61 @@ public abstract class IndexReader {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Returns an IndexReader reading the index in an FSDirectory in the named
|
/** Returns a read/write IndexReader reading the index in an FSDirectory in the named
|
||||||
path.
|
path. <b>NOTE</b>: starting in 3.0 this will return a readOnly IndexReader.
|
||||||
* @throws CorruptIndexException if the index is corrupt
|
* @throws CorruptIndexException if the index is corrupt
|
||||||
* @throws IOException if there is a low-level IO error
|
* @throws IOException if there is a low-level IO error
|
||||||
* @param path the path to the index directory */
|
* @param path the path to the index directory */
|
||||||
public static IndexReader open(String path) throws CorruptIndexException, IOException {
|
public static IndexReader open(String path) throws CorruptIndexException, IOException {
|
||||||
return open(FSDirectory.getDirectory(path), true, null, null);
|
return open(FSDirectory.getDirectory(path), true, null, null, READ_ONLY_DEFAULT);
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Returns an IndexReader reading the index in an FSDirectory in the named
|
/** Returns a read/write IndexReader reading the index in an FSDirectory in the named
|
||||||
* path.
|
* path. <b>NOTE</b>: starting in 3.0 this will return a readOnly IndexReader.
|
||||||
* @param path the path to the index directory
|
* @param path the path to the index directory
|
||||||
* @throws CorruptIndexException if the index is corrupt
|
* @throws CorruptIndexException if the index is corrupt
|
||||||
* @throws IOException if there is a low-level IO error
|
* @throws IOException if there is a low-level IO error
|
||||||
*/
|
*/
|
||||||
public static IndexReader open(File path) throws CorruptIndexException, IOException {
|
public static IndexReader open(File path) throws CorruptIndexException, IOException {
|
||||||
return open(FSDirectory.getDirectory(path), true, null, null);
|
return open(FSDirectory.getDirectory(path), true, null, null, READ_ONLY_DEFAULT);
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Returns an IndexReader reading the index in the given Directory.
|
/** Returns a read/write IndexReader reading the index in
|
||||||
|
* the given Directory. <b>NOTE</b>: starting in 3.0 this
|
||||||
|
* will return a readOnly IndexReader.
|
||||||
* @param directory the index directory
|
* @param directory the index directory
|
||||||
* @throws CorruptIndexException if the index is corrupt
|
* @throws CorruptIndexException if the index is corrupt
|
||||||
* @throws IOException if there is a low-level IO error
|
* @throws IOException if there is a low-level IO error
|
||||||
*/
|
*/
|
||||||
public static IndexReader open(final Directory directory) throws CorruptIndexException, IOException {
|
public static IndexReader open(final Directory directory) throws CorruptIndexException, IOException {
|
||||||
return open(directory, false, null, null);
|
return open(directory, false, null, null, READ_ONLY_DEFAULT);
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Expert: returns an IndexReader reading the index in the given
|
/** Returns a read/write or read only IndexReader reading the index in the given Directory.
|
||||||
* {@link IndexCommit}.
|
* @param directory the index directory
|
||||||
|
* @param readOnly true if no changes (deletions, norms) will be made with this IndexReader
|
||||||
|
* @throws CorruptIndexException if the index is corrupt
|
||||||
|
* @throws IOException if there is a low-level IO error
|
||||||
|
*/
|
||||||
|
public static IndexReader open(final Directory directory, boolean readOnly) throws CorruptIndexException, IOException {
|
||||||
|
return open(directory, false, null, null, readOnly);
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Expert: returns a read/write IndexReader reading the index in the given
|
||||||
|
* {@link IndexCommit}. <b>NOTE</b>: starting in 3.0 this
|
||||||
|
* will return a readOnly IndexReader.
|
||||||
* @param commit the commit point to open
|
* @param commit the commit point to open
|
||||||
* @throws CorruptIndexException if the index is corrupt
|
* @throws CorruptIndexException if the index is corrupt
|
||||||
* @throws IOException if there is a low-level IO error
|
* @throws IOException if there is a low-level IO error
|
||||||
*/
|
*/
|
||||||
public static IndexReader open(final IndexCommit commit) throws CorruptIndexException, IOException {
|
public static IndexReader open(final IndexCommit commit) throws CorruptIndexException, IOException {
|
||||||
return open(commit.getDirectory(), false, null, commit);
|
return open(commit.getDirectory(), false, null, commit, READ_ONLY_DEFAULT);
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Expert: returns an IndexReader reading the index in the given
|
/** Expert: returns a read/write IndexReader reading the index in the given
|
||||||
* Directory, with a custom {@link IndexDeletionPolicy}.
|
* Directory, with a custom {@link IndexDeletionPolicy}.
|
||||||
|
* <b>NOTE</b>: starting in 3.0 this will return a
|
||||||
|
* readOnly IndexReader.
|
||||||
* @param directory the index directory
|
* @param directory the index directory
|
||||||
* @param deletionPolicy a custom deletion policy (only used
|
* @param deletionPolicy a custom deletion policy (only used
|
||||||
* if you use this reader to perform deletes or to set
|
* if you use this reader to perform deletes or to set
|
||||||
|
@ -229,11 +260,29 @@ public abstract class IndexReader {
|
||||||
* @throws IOException if there is a low-level IO error
|
* @throws IOException if there is a low-level IO error
|
||||||
*/
|
*/
|
||||||
public static IndexReader open(final Directory directory, IndexDeletionPolicy deletionPolicy) throws CorruptIndexException, IOException {
|
public static IndexReader open(final Directory directory, IndexDeletionPolicy deletionPolicy) throws CorruptIndexException, IOException {
|
||||||
return open(directory, false, deletionPolicy, null);
|
return open(directory, false, deletionPolicy, null, READ_ONLY_DEFAULT);
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Expert: returns an IndexReader reading the index in the given
|
/** Expert: returns a read/write or read only IndexReader reading the index in the given
|
||||||
* Directory, using a specific commit and with a custom {@link IndexDeletionPolicy}.
|
* Directory, with a custom {@link IndexDeletionPolicy}.
|
||||||
|
* <b>NOTE</b>: starting in 3.0 this will return a
|
||||||
|
* readOnly IndexReader.
|
||||||
|
* @param directory the index directory
|
||||||
|
* @param deletionPolicy a custom deletion policy (only used
|
||||||
|
* if you use this reader to perform deletes or to set
|
||||||
|
* norms); see {@link IndexWriter} for details.
|
||||||
|
* @param readOnly true if no changes (deletions, norms) will be made with this IndexReader
|
||||||
|
* @throws CorruptIndexException if the index is corrupt
|
||||||
|
* @throws IOException if there is a low-level IO error
|
||||||
|
*/
|
||||||
|
public static IndexReader open(final Directory directory, IndexDeletionPolicy deletionPolicy, boolean readOnly) throws CorruptIndexException, IOException {
|
||||||
|
return open(directory, false, deletionPolicy, null, readOnly);
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Expert: returns a read/write IndexReader reading the index in the given
|
||||||
|
* Directory, using a specific commit and with a custom
|
||||||
|
* {@link IndexDeletionPolicy}. <b>NOTE</b>: starting in
|
||||||
|
* 3.0 this will return a readOnly IndexReader.
|
||||||
* @param commit the specific {@link IndexCommit} to open;
|
* @param commit the specific {@link IndexCommit} to open;
|
||||||
* see {@link IndexReader#listCommits} to list all commits
|
* see {@link IndexReader#listCommits} to list all commits
|
||||||
* in a directory
|
* in a directory
|
||||||
|
@ -244,11 +293,27 @@ public abstract class IndexReader {
|
||||||
* @throws IOException if there is a low-level IO error
|
* @throws IOException if there is a low-level IO error
|
||||||
*/
|
*/
|
||||||
public static IndexReader open(final IndexCommit commit, IndexDeletionPolicy deletionPolicy) throws CorruptIndexException, IOException {
|
public static IndexReader open(final IndexCommit commit, IndexDeletionPolicy deletionPolicy) throws CorruptIndexException, IOException {
|
||||||
return open(commit.getDirectory(), false, deletionPolicy, commit);
|
return open(commit.getDirectory(), false, deletionPolicy, commit, READ_ONLY_DEFAULT);
|
||||||
}
|
}
|
||||||
|
|
||||||
private static IndexReader open(final Directory directory, final boolean closeDirectory, final IndexDeletionPolicy deletionPolicy, final IndexCommit commit) throws CorruptIndexException, IOException {
|
/** Expert: returns a read/write or read only IndexReader reading the index in the given
|
||||||
return DirectoryIndexReader.open(directory, closeDirectory, deletionPolicy, commit);
|
* Directory, using a specific commit and with a custom {@link IndexDeletionPolicy}.
|
||||||
|
* @param commit the specific {@link IndexCommit} to open;
|
||||||
|
* see {@link IndexReader#listCommits} to list all commits
|
||||||
|
* in a directory
|
||||||
|
* @param deletionPolicy a custom deletion policy (only used
|
||||||
|
* if you use this reader to perform deletes or to set
|
||||||
|
* norms); see {@link IndexWriter} for details.
|
||||||
|
* @param readOnly true if no changes (deletions, norms) will be made with this IndexReader
|
||||||
|
* @throws CorruptIndexException if the index is corrupt
|
||||||
|
* @throws IOException if there is a low-level IO error
|
||||||
|
*/
|
||||||
|
public static IndexReader open(final IndexCommit commit, IndexDeletionPolicy deletionPolicy, boolean readOnly) throws CorruptIndexException, IOException {
|
||||||
|
return open(commit.getDirectory(), false, deletionPolicy, commit, readOnly);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static IndexReader open(final Directory directory, final boolean closeDirectory, final IndexDeletionPolicy deletionPolicy, final IndexCommit commit, final boolean readOnly) throws CorruptIndexException, IOException {
|
||||||
|
return DirectoryIndexReader.open(directory, closeDirectory, deletionPolicy, commit, readOnly);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -637,7 +702,7 @@ public abstract class IndexReader {
|
||||||
* be obtained)
|
* be obtained)
|
||||||
* @throws IOException if there is a low-level IO error
|
* @throws IOException if there is a low-level IO error
|
||||||
*/
|
*/
|
||||||
public final synchronized void setNorm(int doc, String field, byte value)
|
public synchronized void setNorm(int doc, String field, byte value)
|
||||||
throws StaleReaderException, CorruptIndexException, LockObtainFailedException, IOException {
|
throws StaleReaderException, CorruptIndexException, LockObtainFailedException, IOException {
|
||||||
ensureOpen();
|
ensureOpen();
|
||||||
acquireWriteLock();
|
acquireWriteLock();
|
||||||
|
@ -762,7 +827,7 @@ public abstract class IndexReader {
|
||||||
* be obtained)
|
* be obtained)
|
||||||
* @throws IOException if there is a low-level IO error
|
* @throws IOException if there is a low-level IO error
|
||||||
*/
|
*/
|
||||||
public final synchronized void deleteDocument(int docNum) throws StaleReaderException, CorruptIndexException, LockObtainFailedException, IOException {
|
public synchronized void deleteDocument(int docNum) throws StaleReaderException, CorruptIndexException, LockObtainFailedException, IOException {
|
||||||
ensureOpen();
|
ensureOpen();
|
||||||
acquireWriteLock();
|
acquireWriteLock();
|
||||||
hasChanges = true;
|
hasChanges = true;
|
||||||
|
@ -793,7 +858,7 @@ public abstract class IndexReader {
|
||||||
* be obtained)
|
* be obtained)
|
||||||
* @throws IOException if there is a low-level IO error
|
* @throws IOException if there is a low-level IO error
|
||||||
*/
|
*/
|
||||||
public final int deleteDocuments(Term term) throws StaleReaderException, CorruptIndexException, LockObtainFailedException, IOException {
|
public int deleteDocuments(Term term) throws StaleReaderException, CorruptIndexException, LockObtainFailedException, IOException {
|
||||||
ensureOpen();
|
ensureOpen();
|
||||||
TermDocs docs = termDocs(term);
|
TermDocs docs = termDocs(term);
|
||||||
if (docs == null) return 0;
|
if (docs == null) return 0;
|
||||||
|
@ -819,7 +884,7 @@ public abstract class IndexReader {
|
||||||
* @throws CorruptIndexException if the index is corrupt
|
* @throws CorruptIndexException if the index is corrupt
|
||||||
* @throws IOException if there is a low-level IO error
|
* @throws IOException if there is a low-level IO error
|
||||||
*/
|
*/
|
||||||
public final synchronized void undeleteAll() throws StaleReaderException, CorruptIndexException, LockObtainFailedException, IOException {
|
public synchronized void undeleteAll() throws StaleReaderException, CorruptIndexException, LockObtainFailedException, IOException {
|
||||||
ensureOpen();
|
ensureOpen();
|
||||||
acquireWriteLock();
|
acquireWriteLock();
|
||||||
hasChanges = true;
|
hasChanges = true;
|
||||||
|
|
|
@ -3015,7 +3015,7 @@ public class IndexWriter {
|
||||||
try {
|
try {
|
||||||
synchronized(this) {
|
synchronized(this) {
|
||||||
if (segmentInfos.size() == 1){ // add existing index, if any
|
if (segmentInfos.size() == 1){ // add existing index, if any
|
||||||
sReader = SegmentReader.get(segmentInfos.info(0));
|
sReader = SegmentReader.get(true, segmentInfos.info(0));
|
||||||
merger.add(sReader);
|
merger.add(sReader);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -3974,7 +3974,7 @@ public class IndexWriter {
|
||||||
|
|
||||||
for (int i = 0; i < numSegments; i++) {
|
for (int i = 0; i < numSegments; i++) {
|
||||||
SegmentInfo si = sourceSegmentsClone.info(i);
|
SegmentInfo si = sourceSegmentsClone.info(i);
|
||||||
IndexReader reader = SegmentReader.get(si, MERGE_READ_BUFFER_SIZE, merge.mergeDocStores); // no need to set deleter (yet)
|
IndexReader reader = SegmentReader.get(true, si, MERGE_READ_BUFFER_SIZE, merge.mergeDocStores); // no need to set deleter (yet)
|
||||||
merger.add(reader);
|
merger.add(reader);
|
||||||
totDocCount += reader.numDocs();
|
totDocCount += reader.numDocs();
|
||||||
}
|
}
|
||||||
|
|
|
@ -42,8 +42,9 @@ class MultiSegmentReader extends DirectoryIndexReader {
|
||||||
private boolean hasDeletions = false;
|
private boolean hasDeletions = false;
|
||||||
|
|
||||||
/** Construct reading the named set of readers. */
|
/** Construct reading the named set of readers. */
|
||||||
MultiSegmentReader(Directory directory, SegmentInfos sis, boolean closeDirectory) throws IOException {
|
MultiSegmentReader(Directory directory, SegmentInfos sis, boolean closeDirectory, boolean readOnly) throws IOException {
|
||||||
super(directory, sis, closeDirectory);
|
super(directory, sis, closeDirectory, readOnly);
|
||||||
|
|
||||||
// To reduce the chance of hitting FileNotFound
|
// To reduce the chance of hitting FileNotFound
|
||||||
// (and having to retry), we open segments in
|
// (and having to retry), we open segments in
|
||||||
// reverse because IndexWriter merges & deletes
|
// reverse because IndexWriter merges & deletes
|
||||||
|
@ -52,7 +53,7 @@ class MultiSegmentReader extends DirectoryIndexReader {
|
||||||
SegmentReader[] readers = new SegmentReader[sis.size()];
|
SegmentReader[] readers = new SegmentReader[sis.size()];
|
||||||
for (int i = sis.size()-1; i >= 0; i--) {
|
for (int i = sis.size()-1; i >= 0; i--) {
|
||||||
try {
|
try {
|
||||||
readers[i] = SegmentReader.get(sis.info(i));
|
readers[i] = SegmentReader.get(readOnly, sis.info(i));
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
// Close all readers we had opened:
|
// Close all readers we had opened:
|
||||||
for(i++;i<sis.size();i++) {
|
for(i++;i<sis.size();i++) {
|
||||||
|
@ -70,8 +71,8 @@ class MultiSegmentReader extends DirectoryIndexReader {
|
||||||
}
|
}
|
||||||
|
|
||||||
/** This contructor is only used for {@link #reopen()} */
|
/** This contructor is only used for {@link #reopen()} */
|
||||||
MultiSegmentReader(Directory directory, SegmentInfos infos, boolean closeDirectory, SegmentReader[] oldReaders, int[] oldStarts, Map oldNormsCache) throws IOException {
|
MultiSegmentReader(Directory directory, SegmentInfos infos, boolean closeDirectory, SegmentReader[] oldReaders, int[] oldStarts, Map oldNormsCache, boolean readOnly) throws IOException {
|
||||||
super(directory, infos, closeDirectory);
|
super(directory, infos, closeDirectory, readOnly);
|
||||||
|
|
||||||
// we put the old SegmentReaders in a map, that allows us
|
// we put the old SegmentReaders in a map, that allows us
|
||||||
// to lookup a reader using its segment name
|
// to lookup a reader using its segment name
|
||||||
|
@ -106,7 +107,7 @@ class MultiSegmentReader extends DirectoryIndexReader {
|
||||||
SegmentReader newReader;
|
SegmentReader newReader;
|
||||||
if (newReaders[i] == null || infos.info(i).getUseCompoundFile() != newReaders[i].getSegmentInfo().getUseCompoundFile()) {
|
if (newReaders[i] == null || infos.info(i).getUseCompoundFile() != newReaders[i].getSegmentInfo().getUseCompoundFile()) {
|
||||||
// this is a new reader; in case we hit an exception we can close it safely
|
// this is a new reader; in case we hit an exception we can close it safely
|
||||||
newReader = SegmentReader.get(infos.info(i));
|
newReader = SegmentReader.get(readOnly, infos.info(i));
|
||||||
} else {
|
} else {
|
||||||
newReader = (SegmentReader) newReaders[i].reopenSegment(infos.info(i));
|
newReader = (SegmentReader) newReaders[i].reopenSegment(infos.info(i));
|
||||||
}
|
}
|
||||||
|
@ -196,11 +197,12 @@ class MultiSegmentReader extends DirectoryIndexReader {
|
||||||
protected synchronized DirectoryIndexReader doReopen(SegmentInfos infos) throws CorruptIndexException, IOException {
|
protected synchronized DirectoryIndexReader doReopen(SegmentInfos infos) throws CorruptIndexException, IOException {
|
||||||
if (infos.size() == 1) {
|
if (infos.size() == 1) {
|
||||||
// The index has only one segment now, so we can't refresh the MultiSegmentReader.
|
// The index has only one segment now, so we can't refresh the MultiSegmentReader.
|
||||||
// Return a new SegmentReader instead
|
// Return a new [ReadOnly]SegmentReader instead
|
||||||
SegmentReader newReader = SegmentReader.get(infos, infos.info(0), false);
|
return SegmentReader.get(readOnly, infos, infos.info(0), false);
|
||||||
return newReader;
|
} else if (readOnly) {
|
||||||
|
return new ReadOnlyMultiSegmentReader(directory, infos, closeDirectory, subReaders, starts, normsCache);
|
||||||
} else {
|
} else {
|
||||||
return new MultiSegmentReader(directory, infos, closeDirectory, subReaders, starts, normsCache);
|
return new MultiSegmentReader(directory, infos, closeDirectory, subReaders, starts, normsCache, false);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -259,7 +261,7 @@ class MultiSegmentReader extends DirectoryIndexReader {
|
||||||
|
|
||||||
public boolean isDeleted(int n) {
|
public boolean isDeleted(int n) {
|
||||||
// Don't call ensureOpen() here (it could affect performance)
|
// Don't call ensureOpen() here (it could affect performance)
|
||||||
int i = readerIndex(n); // find segment num
|
final int i = readerIndex(n); // find segment num
|
||||||
return subReaders[i].isDeleted(n - starts[i]); // dispatch to segment reader
|
return subReaders[i].isDeleted(n - starts[i]); // dispatch to segment reader
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -287,7 +289,7 @@ class MultiSegmentReader extends DirectoryIndexReader {
|
||||||
return readerIndex(n, this.starts, this.subReaders.length);
|
return readerIndex(n, this.starts, this.subReaders.length);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int readerIndex(int n, int[] starts, int numSubReaders) { // find reader for doc n:
|
final static int readerIndex(int n, int[] starts, int numSubReaders) { // find reader for doc n:
|
||||||
int lo = 0; // search starts array
|
int lo = 0; // search starts array
|
||||||
int hi = numSubReaders - 1; // for first element less
|
int hi = numSubReaders - 1; // for first element less
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,37 @@
|
||||||
|
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 org.apache.lucene.store.Directory;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
class ReadOnlyMultiSegmentReader extends MultiSegmentReader {
|
||||||
|
ReadOnlyMultiSegmentReader(Directory directory, SegmentInfos sis, boolean closeDirectory) throws IOException {
|
||||||
|
super(directory, sis, closeDirectory, true);
|
||||||
|
}
|
||||||
|
|
||||||
|
ReadOnlyMultiSegmentReader(Directory directory, SegmentInfos infos, boolean closeDirectory, SegmentReader[] oldReaders, int[] oldStarts, Map oldNormsCache) throws IOException {
|
||||||
|
super(directory, infos, closeDirectory, oldReaders, oldStarts, oldNormsCache, true);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected void acquireWriteLock() {
|
||||||
|
ReadOnlySegmentReader.noWrite();
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,34 @@
|
||||||
|
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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
class ReadOnlySegmentReader extends SegmentReader {
|
||||||
|
|
||||||
|
static void noWrite() {
|
||||||
|
throw new UnsupportedOperationException("This IndexReader cannot make any changes to the index (it was opened with readOnly = true)");
|
||||||
|
}
|
||||||
|
|
||||||
|
protected void acquireWriteLock() {
|
||||||
|
noWrite();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Not synchronized
|
||||||
|
public boolean isDeleted(int n) {
|
||||||
|
return deletedDocs != null && deletedDocs.get(n);
|
||||||
|
}
|
||||||
|
}
|
|
@ -61,6 +61,7 @@ class SegmentReader extends DirectoryIndexReader {
|
||||||
private boolean rollbackNormsDirty = false;
|
private boolean rollbackNormsDirty = false;
|
||||||
private boolean rollbackUndeleteAll = false;
|
private boolean rollbackUndeleteAll = false;
|
||||||
private int rollbackPendingDeleteCount;
|
private int rollbackPendingDeleteCount;
|
||||||
|
private boolean readOnly;
|
||||||
|
|
||||||
IndexInput freqStream;
|
IndexInput freqStream;
|
||||||
IndexInput proxStream;
|
IndexInput proxStream;
|
||||||
|
@ -191,12 +192,38 @@ class SegmentReader extends DirectoryIndexReader {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static Class READONLY_IMPL;
|
||||||
|
static {
|
||||||
|
try {
|
||||||
|
String name =
|
||||||
|
System.getProperty("org.apache.lucene.ReadOnlySegmentReader.class",
|
||||||
|
ReadOnlySegmentReader.class.getName());
|
||||||
|
READONLY_IMPL = Class.forName(name);
|
||||||
|
} catch (ClassNotFoundException e) {
|
||||||
|
throw new RuntimeException("cannot load ReadOnlySegmentReader class: " + e, e);
|
||||||
|
} catch (SecurityException se) {
|
||||||
|
try {
|
||||||
|
READONLY_IMPL = Class.forName(ReadOnlySegmentReader.class.getName());
|
||||||
|
} catch (ClassNotFoundException e) {
|
||||||
|
throw new RuntimeException("cannot load default ReadOnlySegmentReader class: " + e, e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @throws CorruptIndexException if the index is corrupt
|
* @throws CorruptIndexException if the index is corrupt
|
||||||
* @throws IOException if there is a low-level IO error
|
* @throws IOException if there is a low-level IO error
|
||||||
*/
|
*/
|
||||||
public static SegmentReader get(SegmentInfo si) throws CorruptIndexException, IOException {
|
public static SegmentReader get(SegmentInfo si) throws CorruptIndexException, IOException {
|
||||||
return get(si.dir, si, null, false, false, BufferedIndexInput.BUFFER_SIZE, true);
|
return get(READ_ONLY_DEFAULT, si.dir, si, null, false, false, BufferedIndexInput.BUFFER_SIZE, true);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @throws CorruptIndexException if the index is corrupt
|
||||||
|
* @throws IOException if there is a low-level IO error
|
||||||
|
*/
|
||||||
|
public static SegmentReader get(boolean readOnly, SegmentInfo si) throws CorruptIndexException, IOException {
|
||||||
|
return get(readOnly, si.dir, si, null, false, false, BufferedIndexInput.BUFFER_SIZE, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -204,7 +231,7 @@ class SegmentReader extends DirectoryIndexReader {
|
||||||
* @throws IOException if there is a low-level IO error
|
* @throws IOException if there is a low-level IO error
|
||||||
*/
|
*/
|
||||||
static SegmentReader get(SegmentInfo si, boolean doOpenStores) throws CorruptIndexException, IOException {
|
static SegmentReader get(SegmentInfo si, boolean doOpenStores) throws CorruptIndexException, IOException {
|
||||||
return get(si.dir, si, null, false, false, BufferedIndexInput.BUFFER_SIZE, doOpenStores);
|
return get(READ_ONLY_DEFAULT, si.dir, si, null, false, false, BufferedIndexInput.BUFFER_SIZE, doOpenStores);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -212,7 +239,7 @@ class SegmentReader extends DirectoryIndexReader {
|
||||||
* @throws IOException if there is a low-level IO error
|
* @throws IOException if there is a low-level IO error
|
||||||
*/
|
*/
|
||||||
public static SegmentReader get(SegmentInfo si, int readBufferSize) throws CorruptIndexException, IOException {
|
public static SegmentReader get(SegmentInfo si, int readBufferSize) throws CorruptIndexException, IOException {
|
||||||
return get(si.dir, si, null, false, false, readBufferSize, true);
|
return get(READ_ONLY_DEFAULT, si.dir, si, null, false, false, readBufferSize, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -220,16 +247,24 @@ class SegmentReader extends DirectoryIndexReader {
|
||||||
* @throws IOException if there is a low-level IO error
|
* @throws IOException if there is a low-level IO error
|
||||||
*/
|
*/
|
||||||
static SegmentReader get(SegmentInfo si, int readBufferSize, boolean doOpenStores) throws CorruptIndexException, IOException {
|
static SegmentReader get(SegmentInfo si, int readBufferSize, boolean doOpenStores) throws CorruptIndexException, IOException {
|
||||||
return get(si.dir, si, null, false, false, readBufferSize, doOpenStores);
|
return get(READ_ONLY_DEFAULT, si.dir, si, null, false, false, readBufferSize, doOpenStores);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @throws CorruptIndexException if the index is corrupt
|
* @throws CorruptIndexException if the index is corrupt
|
||||||
* @throws IOException if there is a low-level IO error
|
* @throws IOException if there is a low-level IO error
|
||||||
*/
|
*/
|
||||||
public static SegmentReader get(SegmentInfos sis, SegmentInfo si,
|
static SegmentReader get(boolean readOnly, SegmentInfo si, int readBufferSize, boolean doOpenStores) throws CorruptIndexException, IOException {
|
||||||
|
return get(readOnly, si.dir, si, null, false, false, readBufferSize, doOpenStores);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @throws CorruptIndexException if the index is corrupt
|
||||||
|
* @throws IOException if there is a low-level IO error
|
||||||
|
*/
|
||||||
|
public static SegmentReader get(boolean readOnly, SegmentInfos sis, SegmentInfo si,
|
||||||
boolean closeDir) throws CorruptIndexException, IOException {
|
boolean closeDir) throws CorruptIndexException, IOException {
|
||||||
return get(si.dir, si, sis, closeDir, true, BufferedIndexInput.BUFFER_SIZE, true);
|
return get(readOnly, si.dir, si, sis, closeDir, true, BufferedIndexInput.BUFFER_SIZE, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -241,14 +276,16 @@ class SegmentReader extends DirectoryIndexReader {
|
||||||
boolean closeDir, boolean ownDir,
|
boolean closeDir, boolean ownDir,
|
||||||
int readBufferSize)
|
int readBufferSize)
|
||||||
throws CorruptIndexException, IOException {
|
throws CorruptIndexException, IOException {
|
||||||
return get(dir, si, sis, closeDir, ownDir, readBufferSize, true);
|
return get(READ_ONLY_DEFAULT, dir, si, sis, closeDir, ownDir, readBufferSize, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @throws CorruptIndexException if the index is corrupt
|
* @throws CorruptIndexException if the index is corrupt
|
||||||
* @throws IOException if there is a low-level IO error
|
* @throws IOException if there is a low-level IO error
|
||||||
*/
|
*/
|
||||||
public static SegmentReader get(Directory dir, SegmentInfo si,
|
public static SegmentReader get(boolean readOnly,
|
||||||
|
Directory dir,
|
||||||
|
SegmentInfo si,
|
||||||
SegmentInfos sis,
|
SegmentInfos sis,
|
||||||
boolean closeDir, boolean ownDir,
|
boolean closeDir, boolean ownDir,
|
||||||
int readBufferSize,
|
int readBufferSize,
|
||||||
|
@ -256,11 +293,14 @@ class SegmentReader extends DirectoryIndexReader {
|
||||||
throws CorruptIndexException, IOException {
|
throws CorruptIndexException, IOException {
|
||||||
SegmentReader instance;
|
SegmentReader instance;
|
||||||
try {
|
try {
|
||||||
|
if (readOnly)
|
||||||
|
instance = (SegmentReader)READONLY_IMPL.newInstance();
|
||||||
|
else
|
||||||
instance = (SegmentReader)IMPL.newInstance();
|
instance = (SegmentReader)IMPL.newInstance();
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
throw new RuntimeException("cannot load SegmentReader class: " + e, e);
|
throw new RuntimeException("cannot load SegmentReader class: " + e, e);
|
||||||
}
|
}
|
||||||
instance.init(dir, sis, closeDir);
|
instance.init(dir, sis, closeDir, readOnly);
|
||||||
instance.initialize(si, readBufferSize, doOpenStores);
|
instance.initialize(si, readBufferSize, doOpenStores);
|
||||||
return instance;
|
return instance;
|
||||||
}
|
}
|
||||||
|
@ -381,10 +421,13 @@ class SegmentReader extends DirectoryIndexReader {
|
||||||
} else {
|
} else {
|
||||||
// segment not referenced anymore, reopen not possible
|
// segment not referenced anymore, reopen not possible
|
||||||
// or segment format changed
|
// or segment format changed
|
||||||
newReader = SegmentReader.get(infos, infos.info(0), false);
|
newReader = SegmentReader.get(readOnly, infos, infos.info(0), false);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
return new MultiSegmentReader(directory, infos, closeDirectory, new SegmentReader[] {this}, null, null);
|
if (readOnly)
|
||||||
|
return new ReadOnlyMultiSegmentReader(directory, infos, closeDirectory, new SegmentReader[] {this}, null, null);
|
||||||
|
else
|
||||||
|
return new MultiSegmentReader(directory, infos, closeDirectory, new SegmentReader[] {this}, null, null, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
return newReader;
|
return newReader;
|
||||||
|
@ -412,9 +455,15 @@ class SegmentReader extends DirectoryIndexReader {
|
||||||
|
|
||||||
|
|
||||||
// clone reader
|
// clone reader
|
||||||
SegmentReader clone = new SegmentReader();
|
SegmentReader clone;
|
||||||
|
if (readOnly)
|
||||||
|
clone = new ReadOnlySegmentReader();
|
||||||
|
else
|
||||||
|
clone = new SegmentReader();
|
||||||
|
|
||||||
boolean success = false;
|
boolean success = false;
|
||||||
try {
|
try {
|
||||||
|
clone.readOnly = readOnly;
|
||||||
clone.directory = directory;
|
clone.directory = directory;
|
||||||
clone.si = si;
|
clone.si = si;
|
||||||
clone.segment = segment;
|
clone.segment = segment;
|
||||||
|
|
|
@ -1329,4 +1329,61 @@ public class TestIndexReader extends LuceneTestCase
|
||||||
r2.close();
|
r2.close();
|
||||||
d.close();
|
d.close();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void testReadOnly() throws Throwable {
|
||||||
|
RAMDirectory d = new MockRAMDirectory();
|
||||||
|
IndexWriter writer = new IndexWriter(d, new StandardAnalyzer(), true, IndexWriter.MaxFieldLength.LIMITED);
|
||||||
|
addDocumentWithFields(writer);
|
||||||
|
writer.commit();
|
||||||
|
addDocumentWithFields(writer);
|
||||||
|
writer.close();
|
||||||
|
|
||||||
|
IndexReader r = IndexReader.open(d, true);
|
||||||
|
try {
|
||||||
|
r.deleteDocument(0);
|
||||||
|
fail();
|
||||||
|
} catch (UnsupportedOperationException uoe) {
|
||||||
|
// expected
|
||||||
|
}
|
||||||
|
|
||||||
|
writer = new IndexWriter(d, new StandardAnalyzer(), false, IndexWriter.MaxFieldLength.LIMITED);
|
||||||
|
addDocumentWithFields(writer);
|
||||||
|
writer.close();
|
||||||
|
|
||||||
|
// Make sure reopen is still readonly:
|
||||||
|
IndexReader r2 = r.reopen();
|
||||||
|
r.close();
|
||||||
|
|
||||||
|
assertFalse(r == r2);
|
||||||
|
|
||||||
|
try {
|
||||||
|
r2.deleteDocument(0);
|
||||||
|
fail();
|
||||||
|
} catch (UnsupportedOperationException uoe) {
|
||||||
|
// expected
|
||||||
|
}
|
||||||
|
|
||||||
|
writer = new IndexWriter(d, new StandardAnalyzer(), false, IndexWriter.MaxFieldLength.LIMITED);
|
||||||
|
writer.optimize();
|
||||||
|
writer.close();
|
||||||
|
|
||||||
|
// Make sure reopen to a single segment is still readonly:
|
||||||
|
IndexReader r3 = r2.reopen();
|
||||||
|
r2.close();
|
||||||
|
|
||||||
|
assertFalse(r == r2);
|
||||||
|
|
||||||
|
try {
|
||||||
|
r3.deleteDocument(0);
|
||||||
|
fail();
|
||||||
|
} catch (UnsupportedOperationException uoe) {
|
||||||
|
// expected
|
||||||
|
}
|
||||||
|
|
||||||
|
// Make sure write lock isn't held
|
||||||
|
writer = new IndexWriter(d, new StandardAnalyzer(), false, IndexWriter.MaxFieldLength.LIMITED);
|
||||||
|
writer.close();
|
||||||
|
|
||||||
|
r3.close();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue