
org.opensearch.index.analysis.NamedAnalyzer Maven / Gradle / Ivy
/*
* SPDX-License-Identifier: Apache-2.0
*
* The OpenSearch Contributors require contributions made to
* this file be licensed under the Apache-2.0 license or a
* compatible open source license.
*/
/*
* Licensed to Elasticsearch under one or more contributor
* license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright
* ownership. Elasticsearch licenses this file to you under
* the Apache License, Version 2.0 (the "License"); you may
* not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/
/*
* Modifications Copyright OpenSearch Contributors. See
* GitHub history for details.
*/
package org.opensearch.index.analysis;
import org.apache.lucene.analysis.Analyzer;
import org.apache.lucene.analysis.DelegatingAnalyzerWrapper;
import org.opensearch.common.annotation.PublicApi;
import org.opensearch.index.mapper.MapperException;
import java.util.ArrayList;
import java.util.List;
import java.util.Objects;
/**
* Named analyzer is an analyzer wrapper around an actual analyzer ({@link #analyzer} that is associated
* with a name ({@link #name()}.
*
* @opensearch.api
*/
@PublicApi(since = "1.0.0")
public class NamedAnalyzer extends DelegatingAnalyzerWrapper {
private final String name;
private final AnalyzerScope scope;
private final Analyzer analyzer;
private final int positionIncrementGap;
private final AnalysisMode analysisMode;
public NamedAnalyzer(NamedAnalyzer analyzer, int positionIncrementGap) {
this(analyzer.name(), analyzer.scope(), analyzer.analyzer(), positionIncrementGap);
}
public NamedAnalyzer(String name, AnalyzerScope scope, Analyzer analyzer) {
this(name, scope, analyzer, Integer.MIN_VALUE);
}
public NamedAnalyzer(String name, AnalyzerScope scope, Analyzer analyzer, int positionIncrementGap) {
super(ERROR_STRATEGY);
this.name = name;
this.scope = scope;
this.analyzer = analyzer;
this.positionIncrementGap = positionIncrementGap;
if (analyzer instanceof org.opensearch.index.analysis.AnalyzerComponentsProvider) {
this.analysisMode = ((org.opensearch.index.analysis.AnalyzerComponentsProvider) analyzer).getComponents().analysisMode();
} else {
this.analysisMode = AnalysisMode.ALL;
}
}
/**
* The name of the analyzer.
*/
public String name() {
return this.name;
}
/**
* The scope of the analyzer.
*/
public AnalyzerScope scope() {
return this.scope;
}
/**
* Returns whether this analyzer can be updated
*/
public AnalysisMode getAnalysisMode() {
return this.analysisMode;
}
/**
* The actual analyzer.
*/
public Analyzer analyzer() {
return this.analyzer;
}
@Override
protected Analyzer getWrappedAnalyzer(String fieldName) {
return this.analyzer;
}
@Override
public int getPositionIncrementGap(String fieldName) {
if (positionIncrementGap != Integer.MIN_VALUE) {
return positionIncrementGap;
}
return super.getPositionIncrementGap(fieldName);
}
/**
* Checks the wrapped analyzer for the provided restricted {@link AnalysisMode} and throws
* an error if the analyzer is not allowed to run in that mode. The error contains more detailed information about
* the offending filters that caused the analyzer to not be allowed in this mode.
*/
public void checkAllowedInMode(AnalysisMode mode) {
Objects.requireNonNull(mode);
if (this.getAnalysisMode() == AnalysisMode.ALL) {
return; // everything allowed if this analyzer is in ALL mode
}
if (this.getAnalysisMode() != mode) {
if (analyzer instanceof AnalyzerComponentsProvider) {
TokenFilterFactory[] tokenFilters = ((AnalyzerComponentsProvider) analyzer).getComponents().getTokenFilters();
List offendingFilters = new ArrayList<>();
for (TokenFilterFactory tokenFilter : tokenFilters) {
AnalysisMode filterMode = tokenFilter.getAnalysisMode();
if (filterMode != AnalysisMode.ALL && filterMode != mode) {
offendingFilters.add(tokenFilter.name());
}
}
throw new MapperException(
"analyzer ["
+ name
+ "] contains filters "
+ offendingFilters
+ " that are not allowed to run in "
+ mode.getReadableName()
+ " mode."
);
} else {
throw new MapperException(
"analyzer [" + name + "] contains components that are not allowed to run in " + mode.getReadableName() + " mode."
);
}
}
}
@Override
public String toString() {
return "analyzer name[" + name + "], analyzer [" + analyzer + "], analysisMode [" + analysisMode + "]";
}
/** It is an error if this is ever used, it means we screwed up! */
static final ReuseStrategy ERROR_STRATEGY = new Analyzer.ReuseStrategy() {
@Override
public TokenStreamComponents getReusableComponents(Analyzer a, String f) {
throw new IllegalStateException("NamedAnalyzer cannot be wrapped with a wrapper, only a delegator");
}
@Override
public void setReusableComponents(Analyzer a, String f, TokenStreamComponents c) {
throw new IllegalStateException("NamedAnalyzer cannot be wrapped with a wrapper, only a delegator");
}
};
@Override
public boolean equals(Object o) {
if (this == o) return true;
if (!(o instanceof NamedAnalyzer)) return false;
NamedAnalyzer that = (NamedAnalyzer) o;
return Objects.equals(name, that.name);
}
@Override
public int hashCode() {
return Objects.hash(name);
}
@Override
public void close() {
super.close();
if (scope == AnalyzerScope.INDEX) {
analyzer.close();
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy