SOLR-9310: fixing concurrency issue and taking care of negative versions

This commit is contained in:
Noble Paul 2016-08-22 23:17:39 +05:30
parent 80f9167807
commit 37ae065591
2 changed files with 11 additions and 17 deletions

View File

@ -33,6 +33,7 @@ import java.util.LinkedList;
import java.util.List; import java.util.List;
import java.util.Map; import java.util.Map;
import java.util.Set; import java.util.Set;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.TimeUnit; import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicLong; import java.util.concurrent.atomic.AtomicLong;
@ -147,8 +148,7 @@ public class SolrIndexSearcher extends IndexSearcher implements Closeable, SolrI
private final String path; private final String path;
private boolean releaseDirectory; private boolean releaseDirectory;
private volatile Map<Long,IndexFingerprint> maxVersionFingerprintCache = new HashMap<>(); private final Map<Long, IndexFingerprint> maxVersionFingerprintCache = new ConcurrentHashMap<>();
private final Object fingerprintLock = new Object();
private static DirectoryReader getReader(SolrCore core, SolrIndexConfig config, DirectoryFactory directoryFactory, private static DirectoryReader getReader(SolrCore core, SolrIndexConfig config, DirectoryFactory directoryFactory,
String path) throws IOException { String path) throws IOException {
@ -2379,22 +2379,16 @@ public class SolrIndexSearcher extends IndexSearcher implements Closeable, SolrI
* gets a cached version of the IndexFingerprint for this searcher * gets a cached version of the IndexFingerprint for this searcher
**/ **/
public IndexFingerprint getIndexFingerprint(long maxVersion) throws IOException { public IndexFingerprint getIndexFingerprint(long maxVersion) throws IOException {
IndexFingerprint fingerprint = maxVersionFingerprintCache.get(maxVersion);
if (fingerprint != null) return fingerprint;
// possibly expensive, so prevent more than one thread from calculating it for this searcher // possibly expensive, so prevent more than one thread from calculating it for this searcher
synchronized (maxVersionFingerprintCache) {
// TODO what happens if updates came out of order, would cached fingerprint still be valid? fingerprint = maxVersionFingerprintCache.get(maxVersionFingerprintCache);
// May be caching fingerprint may lead more problems if (fingerprint != null) return fingerprint;
IndexFingerprint fingerprint = maxVersionFingerprintCache.get(maxVersionFingerprintCache); fingerprint = IndexFingerprint.getFingerprint(this, maxVersion);
if(fingerprint == null) { maxVersionFingerprintCache.put(maxVersion, fingerprint);
synchronized (fingerprintLock) { return fingerprint;
if (maxVersionFingerprintCache.get(maxVersionFingerprintCache) == null) {
log.info("Fingerprint for max version: " + maxVersion + " not found in cache" );
maxVersionFingerprintCache.put(maxVersion, IndexFingerprint.getFingerprint(this, maxVersion));
}
fingerprint = maxVersionFingerprintCache.get(maxVersion) ;
}
} }
return fingerprint;
} }
///////////////////////////////////////////////////////////////////// /////////////////////////////////////////////////////////////////////

View File

@ -952,7 +952,7 @@ public class UpdateLog implements PluginInfoInitialized {
for (List<Update> singleList : updateList) { for (List<Update> singleList : updateList) {
for (Update ptr : singleList) { for (Update ptr : singleList) {
if(ptr.version > maxVersion) continue; if(Math.abs(ptr.version) > Math.abs(maxVersion)) continue;
ret.add(ptr.version); ret.add(ptr.version);
if (--n <= 0) return ret; if (--n <= 0) return ret;
} }