mirror of https://github.com/apache/lucene.git
LUCENE-710 followup: code cosmetics and added documentation.
git-svn-id: https://svn.apache.org/repos/asf/lucene/java/trunk@518734 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
parent
a6d56d5eb9
commit
b04a15016d
|
@ -18,24 +18,37 @@ package org.apache.lucene.index;
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Represents a single commit into an index as seen by the
|
* <p>Expert: represents a single commit into an index as seen by the
|
||||||
* {@link IndexDeletionPolicy}.
|
* {@link IndexDeletionPolicy}.
|
||||||
|
* <p>
|
||||||
|
* Changes to the content of an index are made visible only
|
||||||
|
* after the writer who made that change had written to the
|
||||||
|
* directory a new segments file (<code>segments_N</code>). This point in
|
||||||
|
* time, when the action of writing of a new segments file to the
|
||||||
|
* directory is completed, is therefore an index commit point.
|
||||||
|
* <p>
|
||||||
|
* Each index commit point has a unique segments file associated
|
||||||
|
* with it. The segments file associated with a later
|
||||||
|
* index commit point would have a larger N.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
public interface IndexCommitPoint {
|
public interface IndexCommitPoint {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the segments file (ie, <code>segments_N</code>) of
|
* Get the segments file (<code>segments_N</code>) associated
|
||||||
* this commit point.
|
* with this commit point.
|
||||||
*/
|
*/
|
||||||
public String getSegmentsFileName();
|
public String getSegmentsFileName();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Notify the writer that this commit point should be
|
* Delete this commit point.
|
||||||
* deleted. This should only be called by the {@link
|
* <p>
|
||||||
* IndexDeletionPolicy} during its {@link
|
* Upon calling this, the writer is notified that this commit
|
||||||
* IndexDeletionPolicy#onInit} or {@link
|
* point should be deleted.
|
||||||
* IndexDeletionPolicy#onCommit} method.
|
* <p>
|
||||||
|
* Decision that a commit-point should be deleted is taken by the {@link IndexDeletionPolicy} in effect
|
||||||
|
* and therefore this should only be called by its {@link IndexDeletionPolicy#onInit onInit()} or
|
||||||
|
* {@link IndexDeletionPolicy#onCommit onCommit()} methods.
|
||||||
*/
|
*/
|
||||||
public void delete();
|
public void delete();
|
||||||
}
|
}
|
||||||
|
|
|
@ -21,10 +21,13 @@ import java.util.List;
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* <p>Expert: implement this interface, and pass it to one
|
* <p>Expert: policy for deletion of stale {@link IndexCommitPoint index commits}.
|
||||||
|
*
|
||||||
|
* <p>Implement this interface, and pass it to one
|
||||||
* of the {@link IndexWriter} or {@link IndexReader}
|
* of the {@link IndexWriter} or {@link IndexReader}
|
||||||
* constructors, to customize when "point in time" commits
|
* constructors, to customize when older
|
||||||
* are deleted from an index. The default deletion policy
|
* {@link IndexCommitPoint point-in-time commits}
|
||||||
|
* are deleted from the index directory. The default deletion policy
|
||||||
* is {@link KeepOnlyLastCommitDeletionPolicy}, which always
|
* is {@link KeepOnlyLastCommitDeletionPolicy}, which always
|
||||||
* removes old commits as soon as a new commit is done (this
|
* removes old commits as soon as a new commit is done (this
|
||||||
* matches the behavior before 2.2).</p>
|
* matches the behavior before 2.2).</p>
|
||||||
|
@ -52,31 +55,46 @@ public interface IndexDeletionPolicy {
|
||||||
* instantiated to give the policy a chance to remove old
|
* instantiated to give the policy a chance to remove old
|
||||||
* commit points.</p>
|
* commit points.</p>
|
||||||
*
|
*
|
||||||
* <p>The writer locates all commits present in the index
|
* <p>The writer locates all index commits present in the
|
||||||
* and calls this method. The policy may choose to delete
|
* index directory and calls this method. The policy may
|
||||||
* commit points. To delete a commit point, call the
|
* choose to delete some of the commit points, doing so by
|
||||||
* {@link IndexCommitPoint#delete} method.</p>
|
* calling method {@link IndexCommitPoint#delete delete()}
|
||||||
|
* of {@link IndexCommitPoint}.</p>
|
||||||
*
|
*
|
||||||
* @param commits List of {@link IndexCommitPoint},
|
* <p><u>Note:</u> the last CommitPoint is the most recent one,
|
||||||
|
* i.e. the "front index state". Be careful not to delete it,
|
||||||
|
* unless you know for sure what you are doing, and unless
|
||||||
|
* you can afford to lose the index content while doing that.
|
||||||
|
*
|
||||||
|
* @param commits List of current
|
||||||
|
* {@link IndexCommitPoint point-in-time commits},
|
||||||
* sorted by age (the 0th one is the oldest commit).
|
* sorted by age (the 0th one is the oldest commit).
|
||||||
*/
|
*/
|
||||||
public void onInit(List commits) throws IOException;
|
public void onInit(List commits) throws IOException;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* <p>This is called each time the writer commits. This
|
* <p>This is called each time the writer completed a commit.
|
||||||
* gives the policy a chance to remove old commit points
|
* This gives the policy a chance to remove old commit points
|
||||||
* with each commit.</p>
|
* with each commit.</p>
|
||||||
*
|
*
|
||||||
|
* <p>The policy may now choose to delete old commit points
|
||||||
|
* by calling method {@link IndexCommitPoint#delete delete()}
|
||||||
|
* of {@link IndexCommitPoint}.</p>
|
||||||
|
*
|
||||||
* <p>If writer has <code>autoCommit = true</code> then
|
* <p>If writer has <code>autoCommit = true</code> then
|
||||||
* this method will in general be called many times during
|
* this method will in general be called many times during
|
||||||
* one instance of {@link IndexWriter}. If
|
* one instance of {@link IndexWriter}. If
|
||||||
* <code>autoCommit = false</code> then this method is
|
* <code>autoCommit = false</code> then this method is
|
||||||
* only called once when {@link IndexWriter#close} is
|
* only called once when {@link IndexWriter#close} is
|
||||||
* called, or not at all if the {@link IndexWriter#abort}
|
* called, or not at all if the {@link IndexWriter#abort}
|
||||||
* is called. The policy may now choose to delete old
|
* is called.
|
||||||
* commit points by calling {@link IndexCommitPoint#delete}.
|
|
||||||
*
|
*
|
||||||
* @param commits List of {@link IndexCommitPoint}>,
|
* <p><u>Note:</u> the last CommitPoint is the most recent one,
|
||||||
|
* i.e. the "front index state". Be careful not to delete it,
|
||||||
|
* unless you know for sure what you are doing, and unless
|
||||||
|
* you can afford to lose the index content while doing that.
|
||||||
|
*
|
||||||
|
* @param commits List of {@link IndexCommitPoint},
|
||||||
* sorted by age (the 0th one is the oldest commit).
|
* sorted by age (the 0th one is the oldest commit).
|
||||||
*/
|
*/
|
||||||
public void onCommit(List commits) throws IOException;
|
public void onCommit(List commits) throws IOException;
|
||||||
|
|
|
@ -34,19 +34,30 @@ import java.util.Collections;
|
||||||
/*
|
/*
|
||||||
* This class keeps track of each SegmentInfos instance that
|
* This class keeps track of each SegmentInfos instance that
|
||||||
* is still "live", either because it corresponds to a
|
* is still "live", either because it corresponds to a
|
||||||
* segments_N in the Directory (a real commit) or because
|
* segments_N file in the Directory (a "commit", i.e. a
|
||||||
* it's the in-memory SegmentInfos that a writer is actively
|
* committed SegmentInfos) or because it's the in-memory SegmentInfos
|
||||||
* updating but has not yet committed (currently this only
|
* that a writer is actively updating but has not yet committed
|
||||||
* applies when autoCommit=false in IndexWriter). This
|
* (currently this only applies when autoCommit=false in IndexWriter).
|
||||||
* class uses simple reference counting to map the live
|
* This class uses simple reference counting to map the live
|
||||||
* SegmentInfos instances to individual files in the
|
* SegmentInfos instances to individual files in the Directory.
|
||||||
* Directory.
|
*
|
||||||
|
* The same directory file may be referenced by more than
|
||||||
|
* one IndexCommitPoints, i.e. more than one SegmentInfos.
|
||||||
|
* Therefore we count how many commits reference each file.
|
||||||
|
* When all the commits referencing a certain file have been
|
||||||
|
* deleted, the refcount for that file becomes zero, and the
|
||||||
|
* file is deleted.
|
||||||
*
|
*
|
||||||
* A separate deletion policy interface
|
* A separate deletion policy interface
|
||||||
* (IndexDeletionPolicy) is consulted on creation (onInit)
|
* (IndexDeletionPolicy) is consulted on creation (onInit)
|
||||||
* and once per commit (onCommit), to decide when a commit
|
* and once per commit (onCommit), to decide when a commit
|
||||||
* should be removed.
|
* should be removed.
|
||||||
*
|
*
|
||||||
|
* It is the business of the IndexDeletionPolicy to choose
|
||||||
|
* when to delete commit points. The actual mechanics of
|
||||||
|
* file deletion, retrying, etc, derived from the deletion
|
||||||
|
* of commit points is the business of the IndexFileDeleter.
|
||||||
|
*
|
||||||
* The current default deletion policy is {@link
|
* The current default deletion policy is {@link
|
||||||
* KeepOnlyLastCommitDeletionPolicy}, which removes all
|
* KeepOnlyLastCommitDeletionPolicy}, which removes all
|
||||||
* prior commits when a new commit has completed. This
|
* prior commits when a new commit has completed. This
|
||||||
|
@ -64,8 +75,9 @@ final class IndexFileDeleter {
|
||||||
* so we will retry them again later: */
|
* so we will retry them again later: */
|
||||||
private List deletable;
|
private List deletable;
|
||||||
|
|
||||||
/* Reference count for all files in the index. Maps
|
/* Reference count for all files in the index.
|
||||||
* String to RefCount (class below) instances: */
|
* Counts how many existing commits reference a file.
|
||||||
|
* Maps String to RefCount (class below) instances: */
|
||||||
private Map refCounts = new HashMap();
|
private Map refCounts = new HashMap();
|
||||||
|
|
||||||
/* Holds all commits (segments_N) currently in the index.
|
/* Holds all commits (segments_N) currently in the index.
|
||||||
|
@ -79,8 +91,10 @@ final class IndexFileDeleter {
|
||||||
* non-commit checkpoint: */
|
* non-commit checkpoint: */
|
||||||
private List lastFiles = new ArrayList();
|
private List lastFiles = new ArrayList();
|
||||||
|
|
||||||
|
/* Commits that the IndexDeletionPolicy have decided to delete: */
|
||||||
|
private List commitsToDelete = new ArrayList();
|
||||||
|
|
||||||
private PrintStream infoStream;
|
private PrintStream infoStream;
|
||||||
private List toDelete = new ArrayList();
|
|
||||||
private Directory directory;
|
private Directory directory;
|
||||||
private IndexDeletionPolicy policy;
|
private IndexDeletionPolicy policy;
|
||||||
|
|
||||||
|
@ -188,19 +202,19 @@ final class IndexFileDeleter {
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Remove the CommitPoints in the toDelete List by
|
* Remove the CommitPoints in the commitsToDelete List by
|
||||||
* DecRef'ing all files from each SegmentInfos.
|
* DecRef'ing all files from each SegmentInfos.
|
||||||
*/
|
*/
|
||||||
private void deleteCommits() throws IOException {
|
private void deleteCommits() throws IOException {
|
||||||
|
|
||||||
int size = toDelete.size();
|
int size = commitsToDelete.size();
|
||||||
|
|
||||||
if (size > 0) {
|
if (size > 0) {
|
||||||
|
|
||||||
// First decref all files that had been referred to by
|
// First decref all files that had been referred to by
|
||||||
// the now-deleted commits:
|
// the now-deleted commits:
|
||||||
for(int i=0;i<size;i++) {
|
for(int i=0;i<size;i++) {
|
||||||
CommitPoint commit = (CommitPoint) toDelete.get(i);
|
CommitPoint commit = (CommitPoint) commitsToDelete.get(i);
|
||||||
if (infoStream != null) {
|
if (infoStream != null) {
|
||||||
message("deleteCommits: now remove commit \"" + commit.getSegmentsFileName() + "\"");
|
message("deleteCommits: now remove commit \"" + commit.getSegmentsFileName() + "\"");
|
||||||
}
|
}
|
||||||
|
@ -210,9 +224,9 @@ final class IndexFileDeleter {
|
||||||
}
|
}
|
||||||
decRef(commit.getSegmentsFileName());
|
decRef(commit.getSegmentsFileName());
|
||||||
}
|
}
|
||||||
toDelete.clear();
|
commitsToDelete.clear();
|
||||||
|
|
||||||
// Now compact commits to remove deleted ones:
|
// Now compact commits to remove deleted ones (preserving the sort):
|
||||||
size = commits.size();
|
size = commits.size();
|
||||||
int readFrom = 0;
|
int readFrom = 0;
|
||||||
int writeTo = 0;
|
int writeTo = 0;
|
||||||
|
@ -258,6 +272,9 @@ final class IndexFileDeleter {
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
* For definition of "check point" see IndexWriter comments:
|
||||||
|
* "Clarification: Check Points (and commits)".
|
||||||
|
*
|
||||||
* Writer calls this when it has made a "consistent
|
* Writer calls this when it has made a "consistent
|
||||||
* change" to the index, meaning new files are written to
|
* change" to the index, meaning new files are written to
|
||||||
* the index and the in-memory SegmentInfos have been
|
* the index and the in-memory SegmentInfos have been
|
||||||
|
@ -422,10 +439,10 @@ final class IndexFileDeleter {
|
||||||
public void deleteDirect(Directory otherDir, List segments) throws IOException {
|
public void deleteDirect(Directory otherDir, List segments) throws IOException {
|
||||||
int size = segments.size();
|
int size = segments.size();
|
||||||
for(int i=0;i<size;i++) {
|
for(int i=0;i<size;i++) {
|
||||||
List toDelete = ((SegmentInfo) segments.get(i)).files();
|
List filestoDelete = ((SegmentInfo) segments.get(i)).files();
|
||||||
int size2 = toDelete.size();
|
int size2 = filestoDelete.size();
|
||||||
for(int j=0;j<size2;j++) {
|
for(int j=0;j<size2;j++) {
|
||||||
otherDir.deleteFile((String) toDelete.get(j));
|
otherDir.deleteFile((String) filestoDelete.get(j));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -487,7 +504,7 @@ final class IndexFileDeleter {
|
||||||
public void delete() {
|
public void delete() {
|
||||||
if (!deleted) {
|
if (!deleted) {
|
||||||
deleted = true;
|
deleted = true;
|
||||||
toDelete.add(this);
|
commitsToDelete.add(this);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -44,8 +44,8 @@ final class IndexFileNames {
|
||||||
/** Extension of deletes */
|
/** Extension of deletes */
|
||||||
static final String DELETES_EXTENSION = "del";
|
static final String DELETES_EXTENSION = "del";
|
||||||
|
|
||||||
/** Extension of single norms */
|
/** Extension of plain norms */
|
||||||
static final String SINGLE_NORMS_EXTENSION = "f";
|
static final String PLAIN_NORMS_EXTENSION = "f";
|
||||||
|
|
||||||
/** Extension of separate norms */
|
/** Extension of separate norms */
|
||||||
static final String SEPARATE_NORMS_EXTENSION = "s";
|
static final String SEPARATE_NORMS_EXTENSION = "s";
|
||||||
|
@ -91,9 +91,9 @@ final class IndexFileNames {
|
||||||
* @param gen -- generation
|
* @param gen -- generation
|
||||||
*/
|
*/
|
||||||
static final String fileNameFromGeneration(String base, String extension, long gen) {
|
static final String fileNameFromGeneration(String base, String extension, long gen) {
|
||||||
if (gen == -1) {
|
if (gen == SegmentInfo.NO) {
|
||||||
return null;
|
return null;
|
||||||
} else if (gen == 0) {
|
} else if (gen == SegmentInfo.WITHOUT_GEN) {
|
||||||
return base + extension;
|
return base + extension;
|
||||||
} else {
|
} else {
|
||||||
return base + "_" + Long.toString(gen, Character.MAX_RADIX) + extension;
|
return base + "_" + Long.toString(gen, Character.MAX_RADIX) + extension;
|
||||||
|
|
|
@ -126,6 +126,30 @@ import java.util.Map.Entry;
|
||||||
normally relies on. </p>
|
normally relies on. </p>
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Clarification: Check Points (and commits)
|
||||||
|
* Being able to set autoCommit=false allows IndexWriter to flush and
|
||||||
|
* write new index files to the directory without writing a new segments_N
|
||||||
|
* file which references these new files. It also means that the state of
|
||||||
|
* the in memory SegmentInfos object is different than the most recent
|
||||||
|
* segments_N file written to the directory.
|
||||||
|
*
|
||||||
|
* Each time the SegmentInfos is changed, and matches the (possibly
|
||||||
|
* modified) directory files, we have a new "check point".
|
||||||
|
* If the modified/new SegmentInfos is written to disk - as a new
|
||||||
|
* (generation of) segments_N file - this check point is also an
|
||||||
|
* IndexCommitPoint.
|
||||||
|
*
|
||||||
|
* With autoCommit=true, every checkPoint is also a CommitPoint.
|
||||||
|
* With autoCommit=false, some checkPoints may not be commits.
|
||||||
|
*
|
||||||
|
* A new checkpoint always replaces the previous checkpoint and
|
||||||
|
* becomes the new "front" of the index. This allows the IndexFileDeleter
|
||||||
|
* to delete files that are referenced only by stale checkpoints.
|
||||||
|
* (files that were created since the last commit, but are no longer
|
||||||
|
* referenced by the "front" of the index). For this, IndexFileDeleter
|
||||||
|
* keeps track of the last non commit checkpoint.
|
||||||
|
*/
|
||||||
public class IndexWriter {
|
public class IndexWriter {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -1427,7 +1451,6 @@ public class IndexWriter {
|
||||||
flushRamSegments();
|
flushRamSegments();
|
||||||
|
|
||||||
// 2 copy segment infos and find the highest level from dirs
|
// 2 copy segment infos and find the highest level from dirs
|
||||||
int start = segmentInfos.size();
|
|
||||||
int startUpperBound = minMergeDocs;
|
int startUpperBound = minMergeDocs;
|
||||||
|
|
||||||
boolean success = false;
|
boolean success = false;
|
||||||
|
@ -1656,6 +1679,8 @@ public class IndexWriter {
|
||||||
/**
|
/**
|
||||||
* Flush all in-memory buffered updates (adds and deletes)
|
* Flush all in-memory buffered updates (adds and deletes)
|
||||||
* to the Directory.
|
* to the Directory.
|
||||||
|
* <p>Note: if <code>autoCommit=false</code>, flushed data would still
|
||||||
|
* not be visible to readers, until {@link #close} is called.
|
||||||
* @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
|
||||||
*/
|
*/
|
||||||
|
|
|
@ -25,6 +25,12 @@ import java.util.List;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
|
|
||||||
final class SegmentInfo {
|
final class SegmentInfo {
|
||||||
|
|
||||||
|
static final int NO = -1; // e.g. no norms; no deletes;
|
||||||
|
static final int YES = 1; // e.g. have norms; have deletes;
|
||||||
|
static final int CHECK_DIR = 0; // e.g. must check dir to see if there are norms/deletions
|
||||||
|
static final int WITHOUT_GEN = 0; // a file name that has no GEN in it.
|
||||||
|
|
||||||
public String name; // unique name in dir
|
public String name; // unique name in dir
|
||||||
public int docCount; // number of docs in seg
|
public int docCount; // number of docs in seg
|
||||||
public Directory dir; // where segment resides
|
public Directory dir; // where segment resides
|
||||||
|
@ -32,17 +38,21 @@ final class SegmentInfo {
|
||||||
private boolean preLockless; // true if this is a segments file written before
|
private boolean preLockless; // true if this is a segments file written before
|
||||||
// lock-less commits (2.1)
|
// lock-less commits (2.1)
|
||||||
|
|
||||||
private long delGen; // current generation of del file; -1 if there
|
private long delGen; // current generation of del file; NO if there
|
||||||
// are no deletes; 0 if it's a pre-2.1 segment
|
// are no deletes; CHECK_DIR if it's a pre-2.1 segment
|
||||||
// (and we must check filesystem); 1 or higher if
|
// (and we must check filesystem); YES or higher if
|
||||||
// there are deletes at generation N
|
// there are deletes at generation N
|
||||||
|
|
||||||
private long[] normGen; // current generations of each field's norm file.
|
private long[] normGen; // current generation of each field's norm file.
|
||||||
// If this array is null, we must check filesystem
|
// If this array is null, for lockLess this means no
|
||||||
// when preLockLess is true. Else,
|
// separate norms. For preLockLess this means we must
|
||||||
// there are no separate norms
|
// check filesystem. If this array is not null, its
|
||||||
|
// values mean: NO says this field has no separate
|
||||||
|
// norms; CHECK_DIR says it is a preLockLess segment and
|
||||||
|
// filesystem must be checked; >= YES says this field
|
||||||
|
// has separate norms with the specified generation
|
||||||
|
|
||||||
private byte isCompoundFile; // -1 if it is not; 1 if it is; 0 if it's
|
private byte isCompoundFile; // NO if it is not; YES if it is; CHECK_DIR if it's
|
||||||
// pre-2.1 (ie, must check file system to see
|
// pre-2.1 (ie, must check file system to see
|
||||||
// if <name>.cfs and <name>.nrm exist)
|
// if <name>.cfs and <name>.nrm exist)
|
||||||
|
|
||||||
|
@ -59,15 +69,15 @@ final class SegmentInfo {
|
||||||
this.name = name;
|
this.name = name;
|
||||||
this.docCount = docCount;
|
this.docCount = docCount;
|
||||||
this.dir = dir;
|
this.dir = dir;
|
||||||
delGen = -1;
|
delGen = NO;
|
||||||
isCompoundFile = 0;
|
isCompoundFile = CHECK_DIR;
|
||||||
preLockless = true;
|
preLockless = true;
|
||||||
hasSingleNormFile = false;
|
hasSingleNormFile = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
public SegmentInfo(String name, int docCount, Directory dir, boolean isCompoundFile, boolean hasSingleNormFile) {
|
public SegmentInfo(String name, int docCount, Directory dir, boolean isCompoundFile, boolean hasSingleNormFile) {
|
||||||
this(name, docCount, dir);
|
this(name, docCount, dir);
|
||||||
this.isCompoundFile = (byte) (isCompoundFile ? 1 : -1);
|
this.isCompoundFile = (byte) (isCompoundFile ? YES : NO);
|
||||||
this.hasSingleNormFile = hasSingleNormFile;
|
this.hasSingleNormFile = hasSingleNormFile;
|
||||||
preLockless = false;
|
preLockless = false;
|
||||||
}
|
}
|
||||||
|
@ -112,7 +122,7 @@ final class SegmentInfo {
|
||||||
hasSingleNormFile = false;
|
hasSingleNormFile = false;
|
||||||
}
|
}
|
||||||
int numNormGen = input.readInt();
|
int numNormGen = input.readInt();
|
||||||
if (numNormGen == -1) {
|
if (numNormGen == NO) {
|
||||||
normGen = null;
|
normGen = null;
|
||||||
} else {
|
} else {
|
||||||
normGen = new long[numNormGen];
|
normGen = new long[numNormGen];
|
||||||
|
@ -121,11 +131,11 @@ final class SegmentInfo {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
isCompoundFile = input.readByte();
|
isCompoundFile = input.readByte();
|
||||||
preLockless = isCompoundFile == 0;
|
preLockless = (isCompoundFile == CHECK_DIR);
|
||||||
} else {
|
} else {
|
||||||
delGen = 0;
|
delGen = CHECK_DIR;
|
||||||
normGen = null;
|
normGen = null;
|
||||||
isCompoundFile = 0;
|
isCompoundFile = CHECK_DIR;
|
||||||
preLockless = true;
|
preLockless = true;
|
||||||
hasSingleNormFile = false;
|
hasSingleNormFile = false;
|
||||||
}
|
}
|
||||||
|
@ -138,11 +148,15 @@ final class SegmentInfo {
|
||||||
// norms set against it yet:
|
// norms set against it yet:
|
||||||
normGen = new long[numFields];
|
normGen = new long[numFields];
|
||||||
|
|
||||||
if (!preLockless) {
|
if (preLockless) {
|
||||||
|
// Do nothing: thus leaving normGen[k]==CHECK_DIR (==0), so that later we know
|
||||||
|
// we have to check filesystem for norm files, because this is prelockless.
|
||||||
|
|
||||||
|
} else {
|
||||||
// This is a FORMAT_LOCKLESS segment, which means
|
// This is a FORMAT_LOCKLESS segment, which means
|
||||||
// there are no separate norms:
|
// there are no separate norms:
|
||||||
for(int i=0;i<numFields;i++) {
|
for(int i=0;i<numFields;i++) {
|
||||||
normGen[i] = -1;
|
normGen[i] = NO;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -152,21 +166,21 @@ final class SegmentInfo {
|
||||||
throws IOException {
|
throws IOException {
|
||||||
// Cases:
|
// Cases:
|
||||||
//
|
//
|
||||||
// delGen == -1: this means this segment was written
|
// delGen == NO: this means this segment was written
|
||||||
// by the LOCKLESS code and for certain does not have
|
// by the LOCKLESS code and for certain does not have
|
||||||
// deletions yet
|
// deletions yet
|
||||||
//
|
//
|
||||||
// delGen == 0: this means this segment was written by
|
// delGen == CHECK_DIR: this means this segment was written by
|
||||||
// pre-LOCKLESS code which means we must check
|
// pre-LOCKLESS code which means we must check
|
||||||
// directory to see if .del file exists
|
// directory to see if .del file exists
|
||||||
//
|
//
|
||||||
// delGen > 0: this means this segment was written by
|
// delGen >= YES: this means this segment was written by
|
||||||
// the LOCKLESS code and for certain has
|
// the LOCKLESS code and for certain has
|
||||||
// deletions
|
// deletions
|
||||||
//
|
//
|
||||||
if (delGen == -1) {
|
if (delGen == NO) {
|
||||||
return false;
|
return false;
|
||||||
} else if (delGen > 0) {
|
} else if (delGen >= YES) {
|
||||||
return true;
|
return true;
|
||||||
} else {
|
} else {
|
||||||
return dir.fileExists(getDelFileName());
|
return dir.fileExists(getDelFileName());
|
||||||
|
@ -175,8 +189,8 @@ final class SegmentInfo {
|
||||||
|
|
||||||
void advanceDelGen() {
|
void advanceDelGen() {
|
||||||
// delGen 0 is reserved for pre-LOCKLESS format
|
// delGen 0 is reserved for pre-LOCKLESS format
|
||||||
if (delGen == -1) {
|
if (delGen == NO) {
|
||||||
delGen = 1;
|
delGen = YES;
|
||||||
} else {
|
} else {
|
||||||
delGen++;
|
delGen++;
|
||||||
}
|
}
|
||||||
|
@ -184,7 +198,7 @@ final class SegmentInfo {
|
||||||
}
|
}
|
||||||
|
|
||||||
void clearDelGen() {
|
void clearDelGen() {
|
||||||
delGen = -1;
|
delGen = NO;
|
||||||
files = null;
|
files = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -201,12 +215,12 @@ final class SegmentInfo {
|
||||||
}
|
}
|
||||||
|
|
||||||
String getDelFileName() {
|
String getDelFileName() {
|
||||||
if (delGen == -1) {
|
if (delGen == NO) {
|
||||||
// In this case we know there is no deletion filename
|
// In this case we know there is no deletion filename
|
||||||
// against this segment
|
// against this segment
|
||||||
return null;
|
return null;
|
||||||
} else {
|
} else {
|
||||||
// If delGen is 0, it's the pre-lockless-commit file format
|
// If delGen is CHECK_DIR, it's the pre-lockless-commit file format
|
||||||
return IndexFileNames.fileNameFromGeneration(name, "." + IndexFileNames.DELETES_EXTENSION, delGen);
|
return IndexFileNames.fileNameFromGeneration(name, "." + IndexFileNames.DELETES_EXTENSION, delGen);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -218,11 +232,11 @@ final class SegmentInfo {
|
||||||
*/
|
*/
|
||||||
boolean hasSeparateNorms(int fieldNumber)
|
boolean hasSeparateNorms(int fieldNumber)
|
||||||
throws IOException {
|
throws IOException {
|
||||||
if ((normGen == null && preLockless) || (normGen != null && normGen[fieldNumber] == 0)) {
|
if ((normGen == null && preLockless) || (normGen != null && normGen[fieldNumber] == CHECK_DIR)) {
|
||||||
// Must fallback to directory file exists check:
|
// Must fallback to directory file exists check:
|
||||||
String fileName = name + ".s" + fieldNumber;
|
String fileName = name + ".s" + fieldNumber;
|
||||||
return dir.fileExists(fileName);
|
return dir.fileExists(fileName);
|
||||||
} else if (normGen == null || normGen[fieldNumber] == -1) {
|
} else if (normGen == null || normGen[fieldNumber] == NO) {
|
||||||
return false;
|
return false;
|
||||||
} else {
|
} else {
|
||||||
return true;
|
return true;
|
||||||
|
@ -258,17 +272,17 @@ final class SegmentInfo {
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// This means this segment was saved with LOCKLESS
|
// This means this segment was saved with LOCKLESS
|
||||||
// code so we first check whether any normGen's are >
|
// code so we first check whether any normGen's are >= 1
|
||||||
// 0 (meaning they definitely have separate norms):
|
// (meaning they definitely have separate norms):
|
||||||
for(int i=0;i<normGen.length;i++) {
|
for(int i=0;i<normGen.length;i++) {
|
||||||
if (normGen[i] > 0) {
|
if (normGen[i] >= YES) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// Next we look for any == 0. These cases were
|
// Next we look for any == 0. These cases were
|
||||||
// pre-LOCKLESS and must be checked in directory:
|
// pre-LOCKLESS and must be checked in directory:
|
||||||
for(int i=0;i<normGen.length;i++) {
|
for(int i=0;i<normGen.length;i++) {
|
||||||
if (normGen[i] == 0) {
|
if (normGen[i] == CHECK_DIR) {
|
||||||
if (hasSeparateNorms(i)) {
|
if (hasSeparateNorms(i)) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
@ -286,8 +300,8 @@ final class SegmentInfo {
|
||||||
* @param fieldIndex field whose norm file will be rewritten
|
* @param fieldIndex field whose norm file will be rewritten
|
||||||
*/
|
*/
|
||||||
void advanceNormGen(int fieldIndex) {
|
void advanceNormGen(int fieldIndex) {
|
||||||
if (normGen[fieldIndex] == -1) {
|
if (normGen[fieldIndex] == NO) {
|
||||||
normGen[fieldIndex] = 1;
|
normGen[fieldIndex] = YES;
|
||||||
} else {
|
} else {
|
||||||
normGen[fieldIndex]++;
|
normGen[fieldIndex]++;
|
||||||
}
|
}
|
||||||
|
@ -304,7 +318,7 @@ final class SegmentInfo {
|
||||||
|
|
||||||
long gen;
|
long gen;
|
||||||
if (normGen == null) {
|
if (normGen == null) {
|
||||||
gen = 0;
|
gen = CHECK_DIR;
|
||||||
} else {
|
} else {
|
||||||
gen = normGen[number];
|
gen = normGen[number];
|
||||||
}
|
}
|
||||||
|
@ -318,12 +332,12 @@ final class SegmentInfo {
|
||||||
if (hasSingleNormFile) {
|
if (hasSingleNormFile) {
|
||||||
// case 2: lockless (or nrm file exists) - single file for all norms
|
// case 2: lockless (or nrm file exists) - single file for all norms
|
||||||
prefix = "." + IndexFileNames.NORMS_EXTENSION;
|
prefix = "." + IndexFileNames.NORMS_EXTENSION;
|
||||||
return IndexFileNames.fileNameFromGeneration(name, prefix, 0);
|
return IndexFileNames.fileNameFromGeneration(name, prefix, WITHOUT_GEN);
|
||||||
}
|
}
|
||||||
|
|
||||||
// case 3: norm file for each field
|
// case 3: norm file for each field
|
||||||
prefix = ".f";
|
prefix = ".f";
|
||||||
return IndexFileNames.fileNameFromGeneration(name, prefix + number, 0);
|
return IndexFileNames.fileNameFromGeneration(name, prefix + number, WITHOUT_GEN);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -334,9 +348,9 @@ final class SegmentInfo {
|
||||||
*/
|
*/
|
||||||
void setUseCompoundFile(boolean isCompoundFile) {
|
void setUseCompoundFile(boolean isCompoundFile) {
|
||||||
if (isCompoundFile) {
|
if (isCompoundFile) {
|
||||||
this.isCompoundFile = 1;
|
this.isCompoundFile = YES;
|
||||||
} else {
|
} else {
|
||||||
this.isCompoundFile = -1;
|
this.isCompoundFile = NO;
|
||||||
}
|
}
|
||||||
files = null;
|
files = null;
|
||||||
}
|
}
|
||||||
|
@ -346,9 +360,9 @@ final class SegmentInfo {
|
||||||
* file; else, false.
|
* file; else, false.
|
||||||
*/
|
*/
|
||||||
boolean getUseCompoundFile() throws IOException {
|
boolean getUseCompoundFile() throws IOException {
|
||||||
if (isCompoundFile == -1) {
|
if (isCompoundFile == NO) {
|
||||||
return false;
|
return false;
|
||||||
} else if (isCompoundFile == 1) {
|
} else if (isCompoundFile == YES) {
|
||||||
return true;
|
return true;
|
||||||
} else {
|
} else {
|
||||||
return dir.fileExists(name + "." + IndexFileNames.COMPOUND_FILE_EXTENSION);
|
return dir.fileExists(name + "." + IndexFileNames.COMPOUND_FILE_EXTENSION);
|
||||||
|
@ -365,7 +379,7 @@ final class SegmentInfo {
|
||||||
output.writeLong(delGen);
|
output.writeLong(delGen);
|
||||||
output.writeByte((byte) (hasSingleNormFile ? 1:0));
|
output.writeByte((byte) (hasSingleNormFile ? 1:0));
|
||||||
if (normGen == null) {
|
if (normGen == null) {
|
||||||
output.writeInt(-1);
|
output.writeInt(NO);
|
||||||
} else {
|
} else {
|
||||||
output.writeInt(normGen.length);
|
output.writeInt(normGen.length);
|
||||||
for(int j = 0; j < normGen.length; j++) {
|
for(int j = 0; j < normGen.length; j++) {
|
||||||
|
@ -405,33 +419,33 @@ final class SegmentInfo {
|
||||||
}
|
}
|
||||||
|
|
||||||
String delFileName = IndexFileNames.fileNameFromGeneration(name, "." + IndexFileNames.DELETES_EXTENSION, delGen);
|
String delFileName = IndexFileNames.fileNameFromGeneration(name, "." + IndexFileNames.DELETES_EXTENSION, delGen);
|
||||||
if (delFileName != null && (delGen > 0 || dir.fileExists(delFileName))) {
|
if (delFileName != null && (delGen >= YES || dir.fileExists(delFileName))) {
|
||||||
files.add(delFileName);
|
files.add(delFileName);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Careful logic for norms files:
|
// Careful logic for norms files
|
||||||
if (normGen != null) {
|
if (normGen != null) {
|
||||||
for(int i=0;i<normGen.length;i++) {
|
for(int i=0;i<normGen.length;i++) {
|
||||||
long gen = normGen[i];
|
long gen = normGen[i];
|
||||||
if (gen > 0) {
|
if (gen >= YES) {
|
||||||
// Definitely a separate norm file, with generation:
|
// Definitely a separate norm file, with generation:
|
||||||
files.add(IndexFileNames.fileNameFromGeneration(name, "." + IndexFileNames.SEPARATE_NORMS_EXTENSION + i, gen));
|
files.add(IndexFileNames.fileNameFromGeneration(name, "." + IndexFileNames.SEPARATE_NORMS_EXTENSION + i, gen));
|
||||||
} else if (-1 == gen) {
|
} else if (NO == gen) {
|
||||||
// No separate norms but maybe non-separate norms
|
// No separate norms but maybe plain norms
|
||||||
// in the non compound file case:
|
// in the non compound file case:
|
||||||
if (!hasSingleNormFile && !useCompoundFile) {
|
if (!hasSingleNormFile && !useCompoundFile) {
|
||||||
String fileName = name + "." + IndexFileNames.SINGLE_NORMS_EXTENSION + i;
|
String fileName = name + "." + IndexFileNames.PLAIN_NORMS_EXTENSION + i;
|
||||||
if (dir.fileExists(fileName)) {
|
if (dir.fileExists(fileName)) {
|
||||||
files.add(fileName);
|
files.add(fileName);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else if (0 == gen) {
|
} else if (CHECK_DIR == gen) {
|
||||||
// Pre-2.1: we have to check file existence
|
// Pre-2.1: we have to check file existence
|
||||||
String fileName = null;
|
String fileName = null;
|
||||||
if (useCompoundFile) {
|
if (useCompoundFile) {
|
||||||
fileName = name + "." + IndexFileNames.SEPARATE_NORMS_EXTENSION + i;
|
fileName = name + "." + IndexFileNames.SEPARATE_NORMS_EXTENSION + i;
|
||||||
} else if (!hasSingleNormFile) {
|
} else if (!hasSingleNormFile) {
|
||||||
fileName = name + "." + IndexFileNames.SINGLE_NORMS_EXTENSION + i;
|
fileName = name + "." + IndexFileNames.PLAIN_NORMS_EXTENSION + i;
|
||||||
}
|
}
|
||||||
if (fileName != null && dir.fileExists(fileName)) {
|
if (fileName != null && dir.fileExists(fileName)) {
|
||||||
files.add(fileName);
|
files.add(fileName);
|
||||||
|
@ -445,7 +459,7 @@ final class SegmentInfo {
|
||||||
if (useCompoundFile)
|
if (useCompoundFile)
|
||||||
prefix = name + "." + IndexFileNames.SEPARATE_NORMS_EXTENSION;
|
prefix = name + "." + IndexFileNames.SEPARATE_NORMS_EXTENSION;
|
||||||
else
|
else
|
||||||
prefix = name + "." + IndexFileNames.SINGLE_NORMS_EXTENSION;
|
prefix = name + "." + IndexFileNames.PLAIN_NORMS_EXTENSION;
|
||||||
int prefixLength = prefix.length();
|
int prefixLength = prefix.length();
|
||||||
String[] allFiles = dir.list();
|
String[] allFiles = dir.list();
|
||||||
if (allFiles == null)
|
if (allFiles == null)
|
||||||
|
|
Loading…
Reference in New Issue