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

org.hibernate.search.stat.impl.StatisticsImpl 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.stat.impl;

import java.io.FileNotFoundException;
import java.io.IOException;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.atomic.AtomicLong;
import java.util.concurrent.atomic.LongAdder;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReadWriteLock;
import java.util.concurrent.locks.ReentrantReadWriteLock;
import java.util.stream.Collectors;

import org.apache.lucene.index.IndexReader;
import org.apache.lucene.index.Term;
import org.apache.lucene.search.BooleanClause;
import org.apache.lucene.search.BooleanQuery;
import org.apache.lucene.search.IndexSearcher;
import org.apache.lucene.search.MatchAllDocsQuery;
import org.apache.lucene.search.TermQuery;
import org.apache.lucene.search.TopDocs;
import org.apache.lucene.store.Directory;
import org.hibernate.search.engine.ProjectionConstants;
import org.hibernate.search.exception.SearchException;
import org.hibernate.search.indexes.spi.DirectoryBasedIndexManager;
import org.hibernate.search.indexes.spi.IndexManager;
import org.hibernate.search.spi.IndexedTypeIdentifier;
import org.hibernate.search.engine.Version;
import org.hibernate.search.engine.integration.impl.ExtendedSearchIntegrator;
import org.hibernate.search.engine.service.classloading.spi.ClassLoadingException;
import org.hibernate.search.stat.Statistics;
import org.hibernate.search.stat.spi.StatisticsImplementor;
import org.hibernate.search.util.impl.ClassLoaderHelper;

/**
 * A concurrent implementation of the {@code Statistics} interface.
 *
 * @author Hardy Ferentschik
 * @author Sanne Grinovero
 */
public class StatisticsImpl implements Statistics, StatisticsImplementor {

	// The following four fields need always be updated
	// as a group. We use the exclusiveLock only on reads though!
	// The hot path is on writing, while a read might never happen
	// and we only care about consistent reads, since races during
	// writes won't loose any data.
	private LongAdder searchQueryCount = new LongAdder();
	private LongAdder searchExecutionTotalTime = new LongAdder();
	// Following is not a LongAdder as it's frequently read while producing stats
	private AtomicLong searchExecutionMaxTime = new AtomicLong();
	private volatile String queryExecutionMaxTimeQueryString;

	private LongAdder objectLoadedCount = new LongAdder();
	private LongAdder objectLoadTotalTime = new LongAdder();
	// Following is not a LongAdder as it's frequently read while producing stats
	private AtomicLong objectLoadMaxTime = new AtomicLong();

	private volatile boolean isStatisticsEnabled;

	private final Lock sharedLock;
	private final Lock exclusiveLock;

	private final ExtendedSearchIntegrator extendedIntegrator;

	public StatisticsImpl(ExtendedSearchIntegrator extendedIntegrator) {
		ReadWriteLock lock = new ReentrantReadWriteLock();
		sharedLock = lock.readLock();
		exclusiveLock = lock.writeLock();
		this.extendedIntegrator = extendedIntegrator;
	}

	@Override
	public void clear() {
		searchQueryCount.reset();
		searchExecutionTotalTime.reset();
		searchExecutionMaxTime.set( 0 );
		queryExecutionMaxTimeQueryString = "";
		objectLoadedCount.reset();
		objectLoadMaxTime.set( 0 );
		objectLoadTotalTime.reset();
	}

	@Override
	public long getSearchQueryExecutionCount() {
		return searchQueryCount.longValue();
	}

	@Override
	public long getSearchQueryTotalTime() {
		return searchExecutionTotalTime.longValue();
	}

	@Override
	public long getSearchQueryExecutionMaxTime() {
		return searchExecutionMaxTime.longValue();
	}

	@Override
	public long getSearchQueryExecutionAvgTime() {
		final long searchQueryCountLocal;
		final long searchExecutionTotalTimeLocal;
		exclusiveLock.lock();
		try {
			searchQueryCountLocal = searchQueryCount.longValue();
			searchExecutionTotalTimeLocal = searchExecutionTotalTime.longValue();
		}
		finally {
			exclusiveLock.unlock();
		}
		long avgExecutionTime = 0;
		if ( searchQueryCountLocal > 0 ) {
			avgExecutionTime = searchExecutionTotalTimeLocal / searchQueryCountLocal;
		}
		return avgExecutionTime;
	}

	@Override
	public String getSearchQueryExecutionMaxTimeQueryString() {
		return queryExecutionMaxTimeQueryString;
	}

	@Override
	public void searchExecuted(String searchString, long time) {
		sharedLock.lock();
		try {
			boolean isLongestQuery = false;
			for ( long old = searchExecutionMaxTime.get();
					( time > old ) && ( isLongestQuery = searchExecutionMaxTime.compareAndSet( old, time ) );
					old = searchExecutionMaxTime.get() ) {
				// no-op
			}
			if ( isLongestQuery ) {
				queryExecutionMaxTimeQueryString = searchString;
			}
			searchQueryCount.increment();
			searchExecutionTotalTime.add( time );
		}
		finally {
			sharedLock.unlock();
		}
	}

	@Override
	public long getObjectsLoadedCount() {
		return objectLoadedCount.longValue();
	}

	@Override
	public long getObjectLoadingTotalTime() {
		return objectLoadTotalTime.longValue();
	}

	@Override
	public long getObjectLoadingExecutionMaxTime() {
		return objectLoadMaxTime.longValue();
	}

	@Override
	public long getObjectLoadingExecutionAvgTime() {
		exclusiveLock.lock();
		try {
			long avgLoadingTime = 0;
			final long currentObjectLoadedCount = objectLoadedCount.longValue();
			if ( currentObjectLoadedCount > 0 ) {
				avgLoadingTime = objectLoadTotalTime.longValue() / currentObjectLoadedCount;
			}
			return avgLoadingTime;
		}
		finally {
			exclusiveLock.unlock();
		}
	}

	@Override
	public void objectLoadExecuted(long numberOfObjectsLoaded, long time) {
		sharedLock.lock();
		try {
			for ( long old = objectLoadMaxTime.longValue();
				( time > old ) && ( objectLoadMaxTime.compareAndSet( old, time ) );
				old = objectLoadMaxTime.longValue() ) {
				//no-op
			}
			objectLoadedCount.add( numberOfObjectsLoaded );
			objectLoadTotalTime.add( time );
		}
		finally {
			sharedLock.unlock();
		}
	}

	@Override
	public boolean isStatisticsEnabled() {
		return isStatisticsEnabled;
	}

	@Override
	public void setStatisticsEnabled(boolean b) {
		isStatisticsEnabled = b;
	}

	@Override
	public String getSearchVersion() {
		return Version.getVersionString();
	}

	@Override
	public Set getIndexedClassNames() {
		Set indexedClasses = new HashSet();
		for ( IndexedTypeIdentifier clazz : extendedIntegrator.getIndexBindings().keySet() ) {
			indexedClasses.add( clazz.getName() );
		}
		return indexedClasses;
	}

	@Override
	public int getNumberOfIndexedEntities(String entity) {
		Class clazz = getEntityClass( entity );
		IndexReader indexReader = extendedIntegrator.getIndexReaderAccessor().open( clazz );
		try {
			IndexSearcher searcher = new IndexSearcher( indexReader );
			BooleanQuery boolQuery = new BooleanQuery.Builder()
					.add( new MatchAllDocsQuery(), BooleanClause.Occur.FILTER )
					.add( new TermQuery( new Term( ProjectionConstants.OBJECT_CLASS, entity ) ),
							BooleanClause.Occur.FILTER )
					.build();
			try {
				TopDocs topdocs = searcher.search( boolQuery, 1 );
				return topdocs.totalHits;
			}
			catch (IOException e) {
				throw new SearchException( "Unable to execute count query for entity " + entity, e );
			}
		}
		finally {
			extendedIntegrator.getIndexReaderAccessor().close( indexReader );
		}
	}

	@Override
	public Map indexedEntitiesCount() {
		Map countPerEntity = new HashMap();
		for ( String className : getIndexedClassNames() ) {
			countPerEntity.put( className, getNumberOfIndexedEntities( className ) );
		}
		return countPerEntity;
	}

	private Class getEntityClass(String entity) {
		Class clazz;
		try {
			clazz = ClassLoaderHelper.classForName( entity, extendedIntegrator.getServiceManager() );
		}
		catch (ClassLoadingException e) {
			throw new IllegalArgumentException( entity + "not a indexed entity" );
		}
		return clazz;
	}

	@Override
	public long getIndexSize(String indexName) {
		IndexManager indexManager = extendedIntegrator.getIndexManager( indexName );
		if ( indexManager == null ) {
			throw new IllegalArgumentException( "'" + indexName + "' is not a known index" );
		}
		return getIndexSize( indexManager );
	}

	@Override
	public Map indexSizes() {
		return extendedIntegrator.getIndexManagerHolder().getIndexManagers().stream()
				.collect( Collectors.toMap( IndexManager::getIndexName, this::getIndexSize ) );
	}

	private long getIndexSize(IndexManager indexManager) {
		if ( !( indexManager instanceof DirectoryBasedIndexManager ) ) {
			throw new IllegalArgumentException( "Index '" + indexManager.getIndexName()
					+ "' is not a Lucene index" );
		}

		DirectoryBasedIndexManager directoryBasedIndexManager = (DirectoryBasedIndexManager) indexManager;
		Directory directory = directoryBasedIndexManager.getDirectoryProvider().getDirectory();

		long totalSize = 0l;
		try {
			for ( String fileName : directory.listAll() ) {
				try {
					totalSize += directory.fileLength( fileName );
				}
				catch (FileNotFoundException ignored) {
					// Ignore: the file was probably removed since the call to listAll
				}
			}
		}
		catch (IOException e) {
			throw new SearchException( "Unexpected exception while computing size of index '"
					+ indexManager.getIndexName() + "'", e );
		}

		return totalSize;
	}
}






© 2015 - 2025 Weber Informatics LLC | Privacy Policy