mirror of https://github.com/apache/lucene.git
SOLR-593: getNewestSearcher() to fix inform deadlocks from getSearcher(), fix elevation component not decrementing it's reference
git-svn-id: https://svn.apache.org/repos/asf/lucene/solr/trunk@678418 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
parent
91170a2535
commit
cb8eb4236a
|
@ -686,12 +686,21 @@ public final class SolrCore {
|
||||||
// get it (and it will increment the ref count at the same time)
|
// get it (and it will increment the ref count at the same time)
|
||||||
private RefCounted<SolrIndexSearcher> _searcher;
|
private RefCounted<SolrIndexSearcher> _searcher;
|
||||||
|
|
||||||
|
// All of the open searchers. Don't access this directly.
|
||||||
|
// protected by synchronizing on searcherLock.
|
||||||
|
private final LinkedList<RefCounted<SolrIndexSearcher>> _searchers = new LinkedList<RefCounted<SolrIndexSearcher>>();
|
||||||
|
|
||||||
final ExecutorService searcherExecutor = Executors.newSingleThreadExecutor();
|
final ExecutorService searcherExecutor = Executors.newSingleThreadExecutor();
|
||||||
private int onDeckSearchers; // number of searchers preparing
|
private int onDeckSearchers; // number of searchers preparing
|
||||||
private Object searcherLock = new Object(); // the sync object for the searcher
|
private Object searcherLock = new Object(); // the sync object for the searcher
|
||||||
private final int maxWarmingSearchers; // max number of on-deck searchers allowed
|
private final int maxWarmingSearchers; // max number of on-deck searchers allowed
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Return a registered {@link RefCounted}<{@link SolrIndexSearcher}> with
|
||||||
|
* the reference count incremented. It <b>must</b> be decremented when no longer needed.
|
||||||
|
* This method should not be called from SolrCoreAware.inform() since it can result
|
||||||
|
* in a deadlock if useColdSearcher==false.
|
||||||
|
*/
|
||||||
public RefCounted<SolrIndexSearcher> getSearcher() {
|
public RefCounted<SolrIndexSearcher> getSearcher() {
|
||||||
try {
|
try {
|
||||||
return getSearcher(false,true,null);
|
return getSearcher(false,true,null);
|
||||||
|
@ -701,6 +710,28 @@ public final class SolrCore {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Return the newest {@link RefCounted}<{@link SolrIndexSearcher}> with
|
||||||
|
* the reference count incremented. It <b>must</b> be decremented when no longer needed.
|
||||||
|
* If no searcher is currently open, then if openNew==true a new searcher will be opened,
|
||||||
|
* or null is returned if openNew==false.
|
||||||
|
*/
|
||||||
|
public RefCounted<SolrIndexSearcher> getNewestSearcher(boolean openNew) {
|
||||||
|
synchronized (searcherLock) {
|
||||||
|
if (_searchers.isEmpty()) {
|
||||||
|
if (!openNew) return null;
|
||||||
|
// Not currently implemented since simply calling getSearcher during inform()
|
||||||
|
// can result in a deadlock. Right now, solr always opens a searcher first
|
||||||
|
// before calling inform() anyway, so this should never happen.
|
||||||
|
throw new UnsupportedOperationException();
|
||||||
|
}
|
||||||
|
RefCounted<SolrIndexSearcher> newest = _searchers.getLast();
|
||||||
|
newest.incref();
|
||||||
|
return newest;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get a {@link SolrIndexSearcher} or start the process of creating a new one.
|
* Get a {@link SolrIndexSearcher} or start the process of creating a new one.
|
||||||
* <p>
|
* <p>
|
||||||
|
@ -810,6 +841,7 @@ public final class SolrCore {
|
||||||
|
|
||||||
RefCounted<SolrIndexSearcher> currSearcherHolder=null;
|
RefCounted<SolrIndexSearcher> currSearcherHolder=null;
|
||||||
final RefCounted<SolrIndexSearcher> newSearchHolder=newHolder(newSearcher);
|
final RefCounted<SolrIndexSearcher> newSearchHolder=newHolder(newSearcher);
|
||||||
|
|
||||||
if (returnSearcher) newSearchHolder.incref();
|
if (returnSearcher) newSearchHolder.incref();
|
||||||
|
|
||||||
// a signal to decrement onDeckSearchers if something goes wrong.
|
// a signal to decrement onDeckSearchers if something goes wrong.
|
||||||
|
@ -820,6 +852,8 @@ public final class SolrCore {
|
||||||
|
|
||||||
boolean alreadyRegistered = false;
|
boolean alreadyRegistered = false;
|
||||||
synchronized (searcherLock) {
|
synchronized (searcherLock) {
|
||||||
|
_searchers.add(newSearchHolder);
|
||||||
|
|
||||||
if (_searcher == null) {
|
if (_searcher == null) {
|
||||||
// if there isn't a current searcher then we may
|
// if there isn't a current searcher then we may
|
||||||
// want to register this one before warming is complete instead of waiting.
|
// want to register this one before warming is complete instead of waiting.
|
||||||
|
@ -965,6 +999,15 @@ public final class SolrCore {
|
||||||
RefCounted<SolrIndexSearcher> holder = new RefCounted<SolrIndexSearcher>(newSearcher) {
|
RefCounted<SolrIndexSearcher> holder = new RefCounted<SolrIndexSearcher>(newSearcher) {
|
||||||
public void close() {
|
public void close() {
|
||||||
try {
|
try {
|
||||||
|
synchronized(searcherLock) {
|
||||||
|
// it's possible for someone to get a reference via the _searchers queue
|
||||||
|
// and increment the refcount while RefCounted.close() is being called.
|
||||||
|
// we check the refcount again to see if this has happened and abort the close.
|
||||||
|
// This relies on the RefCounted class allowing close() to be called every
|
||||||
|
// time the counter hits zero.
|
||||||
|
if (refcount.get() > 0) return;
|
||||||
|
_searchers.remove(this);
|
||||||
|
}
|
||||||
resource.close();
|
resource.close();
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
log.severe("Error closing searcher:" + SolrException.toStr(e));
|
log.severe("Error closing searcher:" + SolrException.toStr(e));
|
||||||
|
|
|
@ -61,7 +61,9 @@ import org.apache.solr.core.SolrCore;
|
||||||
import org.apache.solr.schema.FieldType;
|
import org.apache.solr.schema.FieldType;
|
||||||
import org.apache.solr.schema.SchemaField;
|
import org.apache.solr.schema.SchemaField;
|
||||||
import org.apache.solr.search.SortSpec;
|
import org.apache.solr.search.SortSpec;
|
||||||
|
import org.apache.solr.search.SolrIndexSearcher;
|
||||||
import org.apache.solr.util.VersionedFile;
|
import org.apache.solr.util.VersionedFile;
|
||||||
|
import org.apache.solr.util.RefCounted;
|
||||||
import org.apache.solr.util.plugin.SolrCoreAware;
|
import org.apache.solr.util.plugin.SolrCoreAware;
|
||||||
import org.apache.solr.request.SolrQueryRequest;
|
import org.apache.solr.request.SolrQueryRequest;
|
||||||
import org.w3c.dom.Node;
|
import org.w3c.dom.Node;
|
||||||
|
@ -182,8 +184,14 @@ public class QueryElevationComponent extends SearchComponent implements SolrCore
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
// preload the first data
|
// preload the first data
|
||||||
IndexReader reader = core.getSearcher().get().getReader();
|
RefCounted<SolrIndexSearcher> searchHolder = null;
|
||||||
|
try {
|
||||||
|
searchHolder = core.getNewestSearcher(true);
|
||||||
|
IndexReader reader = searchHolder.get().getReader();
|
||||||
getElevationMap( reader, core );
|
getElevationMap( reader, core );
|
||||||
|
} finally {
|
||||||
|
if (searchHolder != null) searchHolder.decref();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue