org.elasticsearch.index.mapper.VersionFieldMapper Maven / Gradle / Ivy
/*
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
* or more contributor license agreements. Licensed under the Elastic License
* 2.0 and the Server Side Public License, v 1; you may not use this file except
* in compliance with, at your election, the Elastic License 2.0 or the Server
* Side Public License, v 1.
*/
package org.elasticsearch.index.mapper;
import org.apache.lucene.document.Field;
import org.apache.lucene.document.NumericDocValuesField;
import org.apache.lucene.search.Query;
import org.elasticsearch.index.fielddata.FieldDataContext;
import org.elasticsearch.index.fielddata.IndexFieldData;
import org.elasticsearch.index.fielddata.IndexNumericFieldData.NumericType;
import org.elasticsearch.index.fielddata.plain.SortedNumericIndexFieldData;
import org.elasticsearch.index.query.QueryShardException;
import org.elasticsearch.index.query.SearchExecutionContext;
import org.elasticsearch.script.field.VersionDocValuesField;
import java.util.Collections;
/** Mapper for the _version field.
*
* This is the field mapper for the monotonically increasing document version. If you are looking for the field that stores semver style
* strings in a sortable binary format, you want VersionStringFieldMapper in the xpack VersionField plugin
*/
public class VersionFieldMapper extends MetadataFieldMapper {
public static final String NAME = "_version";
public static final String CONTENT_TYPE = "_version";
public static final VersionFieldMapper INSTANCE = new VersionFieldMapper();
public static final TypeParser PARSER = new FixedTypeParser(c -> INSTANCE);
static final class VersionFieldType extends MappedFieldType {
public static final VersionFieldType INSTANCE = new VersionFieldType();
private VersionFieldType() {
super(NAME, false, false, true, TextSearchInfo.NONE, Collections.emptyMap());
}
@Override
public String typeName() {
return CONTENT_TYPE;
}
@Override
public Query termQuery(Object value, SearchExecutionContext context) {
throw new QueryShardException(context, "The _version field is not searchable");
}
@Override
public ValueFetcher valueFetcher(SearchExecutionContext context, String format) {
return new DocValueFetcher(docValueFormat(format, null), context.getForField(this, FielddataOperation.SEARCH));
}
@Override
public BlockLoader blockLoader(BlockLoaderContext blContext) {
return new BlockDocValuesReader.LongsBlockLoader(name());
}
@Override
public IndexFieldData.Builder fielddataBuilder(FieldDataContext fieldDataContext) {
failIfNoDocValues();
return new SortedNumericIndexFieldData.Builder(name(), NumericType.LONG, VersionDocValuesField::new, isIndexed());
}
}
private VersionFieldMapper() {
super(VersionFieldType.INSTANCE);
}
@Override
public void preParse(DocumentParserContext context) {
final Field version = versionField();
context.version(version);
context.doc().add(version);
}
public static Field versionField() {
// see InternalEngine.updateVersion to see where the real version value is set
return new NumericDocValuesField(NAME, -1L);
}
@Override
public void postParse(DocumentParserContext context) {
// In the case of nested docs, let's fill nested docs with version=1 so that Lucene doesn't write a Bitset for documents
// that don't have the field. This is consistent with the default value for efficiency.
Field version = context.version();
assert version != null;
for (LuceneDocument doc : context.nonRootDocuments()) {
doc.add(version);
}
}
@Override
protected String contentType() {
return CONTENT_TYPE;
}
@Override
public SourceLoader.SyntheticFieldLoader syntheticFieldLoader() {
return SourceLoader.SyntheticFieldLoader.NOTHING;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy