All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.hibernate.search.indexes.impl.PeriodicRefreshingReaderProvider Maven / Gradle / Ivy

/*
 * Hibernate Search, full-text search for your domain model
 *
 * License: GNU Lesser General Public License (LGPL), version 2.1 or later
 * See the lgpl.txt file in the root directory or .
 */
package org.hibernate.search.indexes.impl;

import java.io.IOException;
import java.util.Map;
import java.util.Properties;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReadWriteLock;
import java.util.concurrent.locks.ReentrantReadWriteLock;

import org.apache.lucene.index.DirectoryReader;
import org.apache.lucene.index.IndexReader;
import org.apache.lucene.store.Directory;
import org.hibernate.search.cfg.Environment;
import org.hibernate.search.exception.AssertionFailure;
import org.hibernate.search.exception.SearchException;
import org.hibernate.search.indexes.spi.DirectoryBasedIndexManager;
import org.hibernate.search.indexes.spi.DirectoryBasedReaderProvider;
import org.hibernate.search.store.DirectoryProvider;
import org.hibernate.search.util.configuration.impl.ConfigurationParseHelper;
import org.hibernate.search.util.impl.Executors;
import org.hibernate.search.util.logging.impl.Log;
import org.hibernate.search.util.logging.impl.LoggerFactory;
import java.lang.invoke.MethodHandles;

/**
 * This ReaderProvider shares IndexReader buffers among threads and
 * will return "recently" refreshed ReaderProvider, which implies
 * query results might represent the state of a slightly out of date index.
 *
 * The definition of "recently" refreshed is defined by configuration properties,
 * so the user can choose how much the index being out of date is acceptable.
 *
 * In exchange of allowing slightly out of date query results, the fact that we
 * can cap the frequency of index reopening events can provide significant
 * performance improvements in some architectures.
 *
 * @author Sanne Grinovero (C) 2016 Red Hat Inc.
 */
public class PeriodicRefreshingReaderProvider implements DirectoryBasedReaderProvider {

	public static final int DEFAULT_ACCEPTABLE_MAX_AGE_MS = 5000;
	private static final Log log = LoggerFactory.make( MethodHandles.lookup() );

	/**
	 * contains all Readers (most current per Directory and all unclosed old readers)
	 */
	protected final Map allReaders = new ConcurrentHashMap();

	/**
	 * contains last updated Reader; protected by lockOnOpenLatest (in the values)
	 */
	protected final Map currentReaders = new ConcurrentHashMap();

	private volatile ScheduledExecutorService scheduledExecutorService;
	private int delay;

	private DirectoryProvider directoryProvider;
	private String indexName;

	@Override
	public IndexReader openIndexReader() {
		log.tracef( "Opening IndexReader for directoryProvider %s", indexName );
		Directory directory = directoryProvider.getDirectory();
		PerDirectoryLatestReader directoryLatestReader = currentReaders.get( directory );
		// might eg happen for FSSlaveDirectoryProvider or for mutable SearchFactory
		if ( directoryLatestReader == null ) {
			directoryLatestReader = createReader( directory );
		}
		return directoryLatestReader.getLatestReader();
	}

	@Override
	public void closeIndexReader(IndexReader reader) {
		if ( reader == null ) {
			return;
		}
		log.tracef( "Closing IndexReader: %s", reader );
		ReaderUsagePair container = allReaders.get( reader );
		container.close(); //virtual
	}

	//overridable method for testability:
	protected DirectoryReader readerFactory(final Directory directory) throws IOException {
		return DirectoryReader.open( directory );
	}

	@Override
	public void initialize(DirectoryBasedIndexManager indexManager, Properties props) {
		this.directoryProvider = indexManager.getDirectoryProvider();
		this.indexName = indexManager.getIndexName();
		// Initialize at least one, don't forget directoryProvider might return a different Directory later
		createReader( directoryProvider.getDirectory() );
		this.scheduledExecutorService = Executors.newScheduledThreadPool( "Periodic IndexReader refreshing task for index " + indexName );
		this.delay = ConfigurationParseHelper.getIntValue(
				props,
				Environment.ASYNC_READER_REFRESH_PERIOD_MS,
				DEFAULT_ACCEPTABLE_MAX_AGE_MS );
		this.scheduledExecutorService.scheduleAtFixedRate(
				new IndexRefreshTask(),
				delay,
				delay,
				TimeUnit.MILLISECONDS
		);
	}

	@Override
	public void stop() {
		if ( scheduledExecutorService != null ) {
			try {
				ScheduledExecutorService executorService = scheduledExecutorService;
				scheduledExecutorService = null;
				executorService.shutdown();
				executorService.awaitTermination( 1, TimeUnit.SECONDS );
			}
			catch (InterruptedException e) {
				log.timedOutWaitingShutdownOfReaderProvider( indexName );
			}
		}
		for ( IndexReader reader : allReaders.keySet() ) {
			ReaderUsagePair usage = allReaders.get( reader );
			usage.close();
		}

		if ( allReaders.size() != 0 ) {
			log.readersNotProperlyClosedInReaderProvider();
		}
	}


	/**
	 * Thread safe creation of PerDirectoryLatestReader.
	 *
	 * @param directory The Lucene directory for which to create the reader.
	 * @return either the cached instance for the specified Directory or a newly created one.
	 * @see HSEARCH-250
	 */
	private synchronized PerDirectoryLatestReader createReader(Directory directory) {
		PerDirectoryLatestReader reader = currentReaders.get( directory );
		if ( reader != null ) {
			return reader;
		}

		try {
			reader = new PerDirectoryLatestReader( directory );
			currentReaders.put( directory, reader );
			return reader;
		}
		catch (IOException e) {
			throw new SearchException( "Unable to open Lucene IndexReader for IndexManager " + this.indexName, e );
		}
	}


	/**
	 * Container for the couple IndexReader,UsageCounter.
	 */
	protected final class ReaderUsagePair {

		public final DirectoryReader reader;
		/**
		 * When reaching 0 (always test on change) the reader should be really
		 * closed and then discarded.
		 * Starts at 2 because:
		 * first usage token is artificial: means "current" is not to be closed (+1)
		 * additionally when creating it will be used (+1)
		 */
		protected final AtomicInteger usageCounter = new AtomicInteger( 2 );

		ReaderUsagePair(DirectoryReader r) {
			reader = r;
		}

		/**
		 * Closes the IndexReader if no other resource is using it
		 * in which case the reference to this container will also be removed.
		 */
		public void close() {
			int refCount = usageCounter.decrementAndGet();
			if ( refCount == 0 ) {
				ReaderUsagePair removed = allReaders.remove( reader );//remove ourself
				try {
					reader.close();
				}
				catch (IOException e) {
					log.unableToCloseLuceneIndexReader( e );
				}
				assert removed != null;
			}
			else if ( refCount < 0 ) {
				//doesn't happen with current code, could help spotting future bugs?
				throw new AssertionFailure(
						"Closing an IndexReader for which you didn't own a lock-token, or somebody else which didn't own closed already."
				);
			}
		}

		@Override
		public String toString() {
			return "Reader:" + this.hashCode() + " ref.count=" + usageCounter.get();
		}

	}

	/**
	 * An instance for each DirectoryProvider, pointing to the "current" ReaderUsagePair
	 * for each DirectoryProvider.
	 */
	protected final class PerDirectoryLatestReader {

		/**
		 * Reference to the most current IndexReader for a DirectoryProvider
		 * guarded by read/write locks rl and wl.
		 */
		private ReaderUsagePair current;

		/**
		 * Read/Write locks to ensure the current IndexReader isn't closed while
		 * it's being grabbed for usage.
		 */
		private final Lock rl;
		private final Lock wl;

		/**
		 * @param directory The Directory for which we manage the IndexReader.
		 *
		 * @throws IOException when the index initialization fails.
		 */
		public PerDirectoryLatestReader(Directory directory) throws IOException {
			ReadWriteLock rwl = new ReentrantReadWriteLock();
			rl = rwl.readLock();
			wl = rwl.writeLock();
			DirectoryReader reader = readerFactory( directory );
			ReaderUsagePair initialPair = new ReaderUsagePair( reader );
			initialPair.usageCounter.set( 1 ); //a token to mark as active (preventing real close).
			wl.lock();
			current = initialPair;
			wl.unlock();//only to ensure safe publication of 'current' to requestors using the lock.
			allReaders.put( reader, initialPair );
		}

		public DirectoryReader getLatestReader() {
			rl.lock();
			ReaderUsagePair readerUsagePair = current;
			readerUsagePair.usageCounter.incrementAndGet();
			rl.unlock();
			return readerUsagePair.reader;
		}

		public synchronized void refreshIfNeeded() {
			final ReaderUsagePair readerUsagePair = current;
			final DirectoryReader beforeUpdateReader = readerUsagePair.reader;
			final DirectoryReader updatedReader;
			try {
				updatedReader = DirectoryReader.openIfChanged( beforeUpdateReader );
			}
			catch (IOException e) {
				throw new SearchException( "Unable to reopen IndexReader", e );
			}
			if ( updatedReader != null ) {
				//In this case we need to promote the updated as "current"
				//and start the lazy closing process of the previous one.
				ReaderUsagePair newPair = new ReaderUsagePair( updatedReader );
				allReaders.put( updatedReader, newPair );
				//Acquire the write-lock, both for visibility reasons and to
				//make sure a client can get it and increment the usage counter atomically.
				try {
					wl.lock();
					current = newPair;
					wl.unlock();
				}
				finally {
					if ( readerUsagePair != null ) {
						readerUsagePair.close();// release a token as it's not the current any more.
					}
				}
			}
		}

	}

	private final class IndexRefreshTask implements Runnable {

		@Override
		public void run() {
			for ( PerDirectoryLatestReader lr : currentReaders.values() ) {
				lr.refreshIfNeeded();
			}
		}

	}

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy