org.hibernate.search.analyzer.impl.LuceneEmbeddedAnalyzerStrategy Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of hibernate-search-engine Show documentation
Show all versions of hibernate-search-engine Show documentation
Core of the Object/Lucene mapper, query engine and index management
/*
* 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.analyzer.impl;
import java.text.ParseException;
import java.util.Collection;
import java.util.Collections;
import java.util.HashMap;
import java.util.Map;
import java.util.stream.Stream;
import org.apache.lucene.analysis.Analyzer;
import org.apache.lucene.analysis.standard.StandardAnalyzer;
import org.apache.lucene.util.Version;
import org.hibernate.search.analyzer.definition.LuceneAnalysisDefinitionProvider;
import org.hibernate.search.analyzer.definition.impl.ChainingLuceneAnalysisDefinitionRegistry;
import org.hibernate.search.analyzer.definition.impl.LuceneAnalysisDefinitionRegistry;
import org.hibernate.search.analyzer.definition.impl.LuceneAnalysisDefinitionRegistryBuilderImpl;
import org.hibernate.search.analyzer.definition.impl.SimpleLuceneAnalysisDefinitionRegistry;
import org.hibernate.search.analyzer.definition.spi.LuceneAnalysisDefinitionSourceService;
import org.hibernate.search.analyzer.spi.AnalyzerReference;
import org.hibernate.search.analyzer.spi.AnalyzerStrategy;
import org.hibernate.search.annotations.AnalyzerDef;
import org.hibernate.search.annotations.NormalizerDef;
import org.hibernate.search.backend.impl.lucene.analysis.HibernateSearchNormalizerWrapper;
import org.hibernate.search.cfg.Environment;
import org.hibernate.search.cfg.spi.SearchConfiguration;
import org.hibernate.search.engine.service.spi.ServiceManager;
import org.hibernate.search.engine.service.spi.ServiceReference;
import org.hibernate.search.exception.SearchException;
import org.hibernate.search.util.StringHelper;
import org.hibernate.search.util.impl.ClassLoaderHelper;
import org.hibernate.search.util.impl.PassThroughAnalyzer;
import org.hibernate.search.util.logging.impl.Log;
import org.hibernate.search.util.logging.impl.LoggerFactory;
import java.lang.invoke.MethodHandles;
/**
* @author Yoann Rodiere
*/
public class LuceneEmbeddedAnalyzerStrategy implements AnalyzerStrategy {
private static final Log log = LoggerFactory.make( MethodHandles.lookup() );
private final ServiceManager serviceManager;
private final SearchConfiguration cfg;
private final Version luceneMatchVersion;
private final SimpleLuceneAnalysisDefinitionRegistry defaultDefinitionRegistry;
public LuceneEmbeddedAnalyzerStrategy(ServiceManager serviceManager, SearchConfiguration cfg) {
this.serviceManager = serviceManager;
this.cfg = cfg;
this.luceneMatchVersion = getLuceneMatchVersion( cfg );
/*
* Make sure to re-create the default definitions with each newly instantiated strategy,
* so that the definition providers can add new definitions between two SearchFactory increments.
* Caching those in a Service, for instance, would prevent that.
*/
this.defaultDefinitionRegistry = createDefaultDefinitionRegistry( cfg );
}
private Version getLuceneMatchVersion(SearchConfiguration cfg) {
final Version version;
String tmp = cfg.getProperty( Environment.LUCENE_MATCH_VERSION );
if ( StringHelper.isEmpty( tmp ) ) {
log.recommendConfiguringLuceneVersion();
version = Environment.DEFAULT_LUCENE_MATCH_VERSION;
}
else {
try {
version = Version.parseLeniently( tmp );
if ( log.isDebugEnabled() ) {
log.debug( "Setting Lucene compatibility to Version " + version );
}
}
catch (IllegalArgumentException e) {
throw log.illegalLuceneVersionFormat( tmp, e.getMessage() );
}
catch (ParseException e) {
throw log.illegalLuceneVersionFormat( tmp, e.getMessage() );
}
}
return version;
}
private SimpleLuceneAnalysisDefinitionRegistry createDefaultDefinitionRegistry(SearchConfiguration cfg) {
final LuceneAnalysisDefinitionProvider definitionsProvider = getLuceneAnalyzerDefinitionProvider();
LuceneAnalysisDefinitionRegistryBuilderImpl builder = new LuceneAnalysisDefinitionRegistryBuilderImpl();
if ( definitionsProvider != null ) {
try {
definitionsProvider.register( builder );
}
catch (SearchException e) { // Do not wrap our own exceptions (from the builder, for instance)
throw e;
}
catch (RuntimeException e) { // Do wrap any other exception
throw log.invalidLuceneAnalyzerDefinitionProvider( definitionsProvider.getClass().getName(), e );
}
}
return builder.build();
}
private LuceneAnalysisDefinitionProvider getLuceneAnalyzerDefinitionProvider() {
// Uses a Service so that integrators can inject alternative Lucene Analyzer definition providers
try ( ServiceReference serviceRef = serviceManager.requestReference( LuceneAnalysisDefinitionSourceService.class ) ) {
return serviceRef.get().getLuceneAnalyzerDefinitionProvider();
}
}
@SuppressWarnings("unchecked")
@Override
public LuceneAnalyzerReference createDefaultAnalyzerReference() {
Class extends Analyzer> analyzerClass;
String analyzerClassName = cfg.getProperty( Environment.ANALYZER_CLASS );
if ( analyzerClassName != null ) {
try {
analyzerClass = ClassLoaderHelper.classForName( analyzerClassName, serviceManager );
}
catch (Exception e) {
// Maybe the string refers to an analyzer definition instead?
return createNamedAnalyzerReference( analyzerClassName );
}
}
else {
analyzerClass = StandardAnalyzer.class;
}
return createLuceneClassAnalyzerReference( analyzerClass );
}
@Override
public LuceneAnalyzerReference createPassThroughAnalyzerReference() {
return new SimpleLuceneAnalyzerReference( PassThroughAnalyzer.INSTANCE );
}
@Override
public Map createProvidedAnalyzerReferences() {
Map references = new HashMap<>();
for ( String defaultAnalyzerName : defaultDefinitionRegistry.getAnalyzerDefinitions().keySet() ) {
NamedLuceneAnalyzerReference reference = createNamedAnalyzerReference( defaultAnalyzerName );
references.put( defaultAnalyzerName, reference );
}
return references;
}
@Override
public LuceneAnalyzerReference createLuceneClassAnalyzerReference(Class> analyzerClass) {
Analyzer analyzer = createAnalyzerInstance( analyzerClass );
return new SimpleLuceneAnalyzerReference( analyzer );
}
@Override
public NamedLuceneAnalyzerReference createNamedAnalyzerReference(String name) {
return new NamedLuceneAnalyzerReference( name );
}
@Override
public Map createProvidedNormalizerReferences() {
Map references = new HashMap<>();
for ( String defaultAnalyzerName : defaultDefinitionRegistry.getNormalizerDefinitions().keySet() ) {
NamedLuceneAnalyzerReference reference = createNamedNormalizerReference( defaultAnalyzerName );
references.put( defaultAnalyzerName, reference );
}
return references;
}
@Override
public LuceneAnalyzerReference createLuceneClassNormalizerReference(Class> analyzerClass) {
Analyzer normalizer = createNormalizerInstance( analyzerClass );
return new SimpleLuceneNormalizerReference( normalizer );
}
@Override
public NamedLuceneAnalyzerReference createNamedNormalizerReference(String name) {
return new NamedLuceneNormalizerReference( name );
}
@Override
public void initializeReferences(Collection analyzerReferences, Map mappingAnalyzerDefinitions,
Collection normalizerReferences, Map mappingNormalizerDefinitions) {
LuceneAnalysisDefinitionRegistry definitionRegistry = createAnalyzerDefinitionRegistry( mappingAnalyzerDefinitions, mappingNormalizerDefinitions );
LuceneAnalyzerBuilder builder = new LuceneAnalyzerBuilder( luceneMatchVersion, serviceManager, definitionRegistry );
Stream.concat( analyzerReferences.stream(), normalizerReferences.stream() )
.forEach( r -> initializeReference( r, definitionRegistry, builder ) );
}
@Override
public ScopedLuceneAnalyzerReference.Builder buildScopedAnalyzerReference(AnalyzerReference initialGlobalAnalyzerReference) {
return new ScopedLuceneAnalyzerReference.Builder(
initialGlobalAnalyzerReference.unwrap( LuceneAnalyzerReference.class ),
Collections.emptyMap() );
}
private LuceneAnalysisDefinitionRegistry createAnalyzerDefinitionRegistry(Map mappingAnalyzerDefinitions,
Map mappingNormalizerDefinitions) {
LuceneAnalysisDefinitionRegistry mappingDefinitionRegistry =
new SimpleLuceneAnalysisDefinitionRegistry( mappingAnalyzerDefinitions, mappingNormalizerDefinitions );
/*
* Make default definitions accessible in the final definition registry.
* This final registry has two scopes:
* - the "local" scope, which contains every definitions from the mapping (see above);
* - the "default"/"global" scope, which contains definitions from the default registry.
*
* When fetching definitions, the "local" scope takes precedence over the "default"/"global" scope.
*
* Note that the default definitions may be different
* each time a new instance of this strategy is created,
* i.e. each time the SearchFactory is "incremented".
* Changes to pre-existing default definitions don't matter if the definitions weren't used,
* and are harmless if they were already used
* (because in that case the reference is already initialized,
* so the new version of the definition will be ignored).
*/
LuceneAnalysisDefinitionRegistry definitionRegistry =
new ChainingLuceneAnalysisDefinitionRegistry( mappingDefinitionRegistry, defaultDefinitionRegistry );
return definitionRegistry;
}
private void initializeReference(AnalyzerReference reference, LuceneAnalysisDefinitionRegistry definitionRegistry,
LuceneAnalyzerBuilder builder) {
if ( reference.is( NamedLuceneAnalyzerReference.class ) ) {
NamedLuceneAnalyzerReference namedReference = reference.unwrap( NamedLuceneAnalyzerReference.class );
if ( !namedReference.isInitialized() ) {
namedReference.initialize( builder );
}
}
}
private Analyzer createAnalyzerInstance(Class> analyzerClass) {
try {
return ClassLoaderHelper.analyzerInstanceFromClass( analyzerClass, luceneMatchVersion );
}
catch (ClassCastException e) {
throw new SearchException( "Lucene analyzer does not extend " + Analyzer.class.getName() + ": " + analyzerClass.getName(), e );
}
catch (Exception e) {
throw new SearchException( "Failed to instantiate lucene analyzer with type " + analyzerClass.getName(), e );
}
}
private Analyzer createNormalizerInstance(Class> analyzerClass) {
Analyzer normalizer = createAnalyzerInstance( analyzerClass );
return new HibernateSearchNormalizerWrapper( normalizer, analyzerClass.getName() );
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy