org.hibernate.search.indexes.impl.IndexManagerGroupHolder 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.util.Properties;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentMap;
import org.apache.lucene.search.similarities.Similarity;
import org.hibernate.search.backend.impl.InternalBackendFactory;
import org.hibernate.search.backend.spi.Backend;
import org.hibernate.search.backend.spi.BackendQueueProcessor;
import org.hibernate.search.cfg.Environment;
import org.hibernate.search.cfg.spi.IndexManagerFactory;
import org.hibernate.search.engine.impl.MutableEntityIndexBinding;
import org.hibernate.search.engine.integration.impl.ExtendedSearchIntegrator;
import org.hibernate.search.engine.service.spi.ServiceManager;
import org.hibernate.search.engine.service.spi.ServiceReference;
import org.hibernate.search.indexes.interceptor.EntityIndexingInterceptor;
import org.hibernate.search.indexes.spi.IndexManager;
import org.hibernate.search.indexes.spi.IndexManagerType;
import org.hibernate.search.spi.IndexedTypeIdentifier;
import org.hibernate.search.spi.WorkerBuildContext;
import org.hibernate.search.util.StringHelper;
import org.hibernate.search.util.configuration.impl.MaskedProperty;
import org.hibernate.search.util.logging.impl.Log;
import org.hibernate.search.util.logging.impl.LoggerFactory;
import java.lang.invoke.MethodHandles;
/**
* Holds references to all index managers representing either:
*
* - The only index manager of a given index (no sharding)
*
- The index managers for each shard of a given index (sharding)
*
*
* @author Yoann Rodiere
*/
public class IndexManagerGroupHolder implements AutoCloseable {
private static final Log log = LoggerFactory.make( MethodHandles.lookup() );
private static final String INDEX_SHARD_ID_SEPARATOR = ".";
private final IndexManagerHolder parentHolder;
private final String indexNameBase;
private final Similarity similarity;
private final IndexManagerType indexManagerType;
private final EntityIndexBinder entityIndexBinder;
private final ConcurrentMap indexManagersRegistry = new ConcurrentHashMap<>();
private final ConcurrentMap backendRegistry = new ConcurrentHashMap<>();
public IndexManagerGroupHolder(IndexManagerHolder parentHolder,
String indexNameBase,
Similarity similarity,
IndexManagerType indexManagerType,
EntityIndexBinder entityIndexBinder) {
super();
this.parentHolder = parentHolder;
this.indexNameBase = indexNameBase;
this.similarity = similarity;
this.indexManagerType = indexManagerType;
this.entityIndexBinder = entityIndexBinder;
}
@Override
public synchronized void close() {
for ( Backend backend : backendRegistry.values() ) {
backend.close();
}
backendRegistry.clear();
for ( IndexManager indexManager : indexManagersRegistry.values() ) {
indexManager.destroy();
}
indexManagersRegistry.clear();
}
public String getIndexNameBase() {
return indexNameBase;
}
public Similarity getSimilarity() {
return similarity;
}
public IndexManagerType getIndexManagerType() {
return indexManagerType;
}
MutableEntityIndexBinding bind(IndexedTypeIdentifier entityType, EntityIndexingInterceptor> interceptor,
WorkerBuildContext buildContext) {
// This will create the binding, but also initialize the indexes as necessary
return entityIndexBinder.bind( this, entityType, interceptor, buildContext );
}
public Backend getOrCreateBackend(String indexManagerName, Properties properties, WorkerBuildContext buildContext) {
String backendName = properties.getProperty( Environment.WORKER_BACKEND );
return getOrCreateBackend( backendName, indexManagerName, properties, buildContext );
}
public Backend getOrCreateBackend(String backendName, String indexManagerName, Properties properties, WorkerBuildContext buildContext) {
if ( backendName == null ) {
/*
* The name may be null (meaning "use the default"), but this is annoying here
* because we use it as a key in a may that doesn't accept null keys.
* Since an empty string also means "use the default", we'll use that instead.
*/
backendName = "";
}
String backendIdentifier = entityIndexBinder.createBackendIdentifier( backendName, indexManagerName );
Backend backend = backendRegistry.get( backendIdentifier );
if ( backend != null ) {
return backend;
}
synchronized (this) {
backend = backendRegistry.get( backendIdentifier );
if ( backend != null ) {
return backend;
}
backend = InternalBackendFactory.createBackend( backendName, indexManagerName, properties, buildContext );
backendRegistry.put( backendIdentifier, backend );
return backend;
}
}
IndexManager getOrCreateIndexManager(String shardName, Properties indexProperties, IndexedTypeIdentifier entityType,
WorkerBuildContext context) {
String indexName = indexNameBase;
if ( shardName != null ) {
indexName += INDEX_SHARD_ID_SEPARATOR + shardName;
}
IndexManager indexManager = indexManagersRegistry.get( indexName );
if ( indexManager != null ) {
indexManager.addContainedEntity( entityType );
return indexManager;
}
synchronized (this) {
indexManager = indexManagersRegistry.get( indexName );
if ( indexManager != null ) {
indexManager.addContainedEntity( entityType );
return indexManager;
}
if ( shardName != null ) {
indexProperties = new MaskedProperty( indexProperties, shardName, indexProperties );
}
indexManager = doCreateIndexManager( indexName, entityType, indexProperties, context );
}
return indexManager;
}
private synchronized IndexManager doCreateIndexManager(String indexName, IndexedTypeIdentifier entityType,
Properties indexProperties, WorkerBuildContext workerBuildContext) {
ExtendedSearchIntegrator activeIntegrator = null;
if ( workerBuildContext == null ) {
/*
* The index manager is being dynamically requested at runtime
* (not during bootstrapping).
* We will need to set to integrator on the indexManager to initialize it
* (see further down).
*/
activeIntegrator = parentHolder.getActiveSearchIntegrator();
workerBuildContext = toWorkerBuildContext( activeIntegrator );
}
// get hold of the index manager factory via the service manager
ServiceManager serviceManager = workerBuildContext.getServiceManager();
// create IndexManager instance via the index manager factory
String indexManagerImplementationName = indexProperties.getProperty( Environment.INDEX_MANAGER_IMPL_NAME );
final IndexManager manager;
try ( ServiceReference indexManagerFactoryRef
= serviceManager.requestReference( IndexManagerFactory.class ) ) {
IndexManagerFactory indexManagerFactory = indexManagerFactoryRef.get();
if ( StringHelper.isEmpty( indexManagerImplementationName ) ) {
manager = indexManagerFactory.createDefaultIndexManager();
}
else {
manager = indexManagerFactory.createIndexManagerByName( indexManagerImplementationName );
}
}
// init the IndexManager
try {
manager.initialize( indexName, indexProperties, similarity, workerBuildContext );
}
catch (Exception e) {
throw log.unableToInitializeIndexManager( indexName, e );
}
indexManagersRegistry.put( indexName, manager );
/*
* During backend creation, it may be needed to create another backend,
* for instance if a backend wants to delegate to another implementation.
* But during backend creation, we don't have a reference to this instance,
* so we cannot call getOrCreateBackend().
* To work around this (and to avoid breaking the BackendFactory API),
* we provide IndexManagerHolder.getGroupHolderByIndexManager(),
* but this requires to first register the groupHolder against the index name.
*/
parentHolder.register( indexName, this );
Backend backend = getOrCreateBackend( indexName, indexProperties, workerBuildContext );
BackendQueueProcessor backendQueueProcessor = backend.createQueueProcessor( manager, workerBuildContext );
parentHolder.register( indexName, manager, backendQueueProcessor );
manager.addContainedEntity( entityType );
if ( activeIntegrator != null ) {
manager.setSearchFactory( activeIntegrator );
}
return manager;
}
private WorkerBuildContext toWorkerBuildContext(ExtendedSearchIntegrator integrator) {
//known implementations of SearchFactory passed are MutableSearchFactory and ImmutableSearchFactory
if ( integrator instanceof WorkerBuildContext ) {
return (WorkerBuildContext) integrator;
}
else {
throw log.assertionFailureCannotCastToWorkerBuilderContext( integrator.getClass() );
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy