Skip to content

Commit

Permalink
Use a mapping snapshot for fetching nested docs
Browse files Browse the repository at this point in the history
This uses the mapping snapshot that we built for the search phase
in elastic#66295 for fetching nested documents. This is simpler to reason about
because the mapping snapshot is immutable.
  • Loading branch information
nik9000 committed Dec 29, 2020
1 parent 68a8347 commit 9d7f497
Show file tree
Hide file tree
Showing 13 changed files with 133 additions and 146 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -238,9 +238,9 @@ public IndexWarmer.TerminationHandle warmReader(final IndexShard indexShard, fin
final MapperService mapperService = indexShard.mapperService();
DocumentMapper docMapper = mapperService.documentMapper();
if (docMapper != null) {
if (docMapper.hasNestedObjects()) {
if (docMapper.mappers().hasNested()) {
warmUp.add(Queries.newNonNestedFilter());
docMapper.getNestedParentMappers().stream().map(ObjectMapper::nestedTypeFilter).forEach(warmUp::add);
docMapper.mappers().getNestedParentMappers().stream().map(ObjectMapper::nestedTypeFilter).forEach(warmUp::add);
}
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -34,10 +34,8 @@
import org.elasticsearch.index.mapper.MapperService.MergeReason;

import java.io.IOException;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.stream.Stream;
Expand Down Expand Up @@ -185,10 +183,6 @@ public IndexFieldMapper IndexFieldMapper() {
return metadataMapper(IndexFieldMapper.class);
}

public boolean hasNestedObjects() {
return mappers().hasNested();
}

public MappingLookup mappers() {
return this.fieldMappers;
}
Expand All @@ -212,91 +206,6 @@ public ParsedDocument createNoopTombstoneDoc(String index, String reason) throws
return parsedDoc;
}

/**
* Given an object path, checks to see if any of its parents are non-nested objects
*/
public boolean hasNonNestedParent(String path) {
ObjectMapper mapper = mappers().objectMappers().get(path);
if (mapper == null) {
return false;
}
while (mapper != null) {
if (mapper.nested().isNested() == false) {
return true;
}
if (path.contains(".") == false) {
return false;
}
path = path.substring(0, path.lastIndexOf("."));
mapper = mappers().objectMappers().get(path);
}
return false;
}

/**
* Returns all nested object mappers
*/
public List<ObjectMapper> getNestedMappers() {
List<ObjectMapper> childMappers = new ArrayList<>();
for (ObjectMapper mapper : mappers().objectMappers().values()) {
if (mapper.nested().isNested() == false) {
continue;
}
childMappers.add(mapper);
}
return childMappers;
}

/**
* Returns all nested object mappers which contain further nested object mappers
*
* Used by BitSetProducerWarmer
*/
public List<ObjectMapper> getNestedParentMappers() {
List<ObjectMapper> parents = new ArrayList<>();
for (ObjectMapper mapper : mappers().objectMappers().values()) {
String nestedParentPath = getNestedParent(mapper.fullPath());
if (nestedParentPath == null) {
continue;
}
ObjectMapper parent = mappers().objectMappers().get(nestedParentPath);
if (parent.nested().isNested()) {
parents.add(parent);
}
}
return parents;
}

/**
* Given a nested object path, returns the path to its nested parent
*
* In particular, if a nested field `foo` contains an object field
* `bar.baz`, then calling this method with `foo.bar.baz` will return
* the path `foo`, skipping over the object-but-not-nested `foo.bar`
*/
public String getNestedParent(String path) {
ObjectMapper mapper = mappers().objectMappers().get(path);
if (mapper == null) {
return null;
}
if (path.contains(".") == false) {
return null;
}
do {
path = path.substring(0, path.lastIndexOf("."));
mapper = mappers().objectMappers().get(path);
if (mapper == null) {
return null;
}
if (mapper.nested().isNested()) {
return path;
}
if (path.contains(".") == false) {
return null;
}
} while(true);
}

public DocumentMapper merge(Mapping mapping, MergeReason reason) {
Mapping merged = this.mapping.merge(mapping, reason);
return new DocumentMapper(this.indexSettings, this.indexAnalyzers, this.documentParser, merged);
Expand All @@ -310,7 +219,7 @@ public void validate(IndexSettings settings, boolean checkLimits) {
+ "required for partitioned index [" + settings.getIndex().getName() + "]");
}
}
if (settings.getIndexSortConfig().hasIndexSort() && hasNestedObjects()) {
if (settings.getIndexSortConfig().hasIndexSort() && mappers().hasNested()) {
throw new IllegalArgumentException("cannot have nested fields when index sort is activated");
}
if (checkLimits) {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -323,4 +323,89 @@ public boolean isSourceEnabled() {
public CacheKey cacheKey() {
return cacheKey;
}

/**
* Given an object path, checks to see if any of its parents are non-nested objects
*/
public boolean hasNonNestedParent(String path) {
ObjectMapper mapper = objectMappers().get(path);
if (mapper == null) {
return false;
}
while (mapper != null) {
if (mapper.nested().isNested() == false) {
return true;
}
if (path.contains(".") == false) {
return false;
}
path = path.substring(0, path.lastIndexOf("."));
mapper = objectMappers().get(path);
}
return false;
}

/**
* Returns all nested object mappers
*/
public List<ObjectMapper> getNestedMappers() {
List<ObjectMapper> childMappers = new ArrayList<>();
for (ObjectMapper mapper : objectMappers().values()) {
if (mapper.nested().isNested() == false) {
continue;
}
childMappers.add(mapper);
}
return childMappers;
}

/**
* Returns all nested object mappers which contain further nested object mappers
*
* Used by BitSetProducerWarmer
*/
public List<ObjectMapper> getNestedParentMappers() {
List<ObjectMapper> parents = new ArrayList<>();
for (ObjectMapper mapper : objectMappers().values()) {
String nestedParentPath = getNestedParent(mapper.fullPath());
if (nestedParentPath == null) {
continue;
}
ObjectMapper parent = objectMappers().get(nestedParentPath);
if (parent.nested().isNested()) {
parents.add(parent);
}
}
return parents;
}

/**
* Given a nested object path, returns the path to its nested parent
*
* In particular, if a nested field `foo` contains an object field
* `bar.baz`, then calling this method with `foo.bar.baz` will return
* the path `foo`, skipping over the object-but-not-nested `foo.bar`
*/
public String getNestedParent(String path) {
ObjectMapper mapper = objectMappers().get(path);
if (mapper == null) {
return null;
}
if (path.contains(".") == false) {
return null;
}
do {
path = path.substring(0, path.lastIndexOf("."));
mapper = objectMappers().get(path);
if (mapper == null) {
return null;
}
if (mapper.nested().isNested()) {
return path;
}
if (path.contains(".") == false) {
return null;
}
} while(true);
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -64,6 +64,7 @@
import org.elasticsearch.script.ScriptContext;
import org.elasticsearch.script.ScriptFactory;
import org.elasticsearch.script.ScriptService;
import org.elasticsearch.search.NestedDocuments;
import org.elasticsearch.search.aggregations.support.ValuesSourceRegistry;
import org.elasticsearch.search.lookup.SearchLookup;
import org.elasticsearch.transport.RemoteClusterAware;
Expand Down Expand Up @@ -628,4 +629,8 @@ private static Map<String, MappedFieldType> parseRuntimeMappings(Map<String, Obj
public MappingLookup.CacheKey mappingCacheKey() {
return mappingLookup.cacheKey();
}

public NestedDocuments getNestedDocuments() {
return new NestedDocuments(mappingLookup, bitsetFilterCache::getBitSetProducer);
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -687,11 +687,6 @@ public QuerySearchResult queryResult() {
return queryResult;
}

@Override
public NestedDocuments getNestedDocuments() {
return new NestedDocuments(indexService.mapperService(), bitsetFilterCache()::getBitSetProducer);
}

@Override
public FetchPhase fetchPhase() {
return fetchPhase;
Expand Down
20 changes: 10 additions & 10 deletions server/src/main/java/org/elasticsearch/search/NestedDocuments.java
Original file line number Diff line number Diff line change
Expand Up @@ -30,7 +30,7 @@
import org.apache.lucene.search.join.BitSetProducer;
import org.apache.lucene.util.BitSet;
import org.elasticsearch.common.lucene.search.Queries;
import org.elasticsearch.index.mapper.MapperService;
import org.elasticsearch.index.mapper.MappingLookup;
import org.elasticsearch.index.mapper.ObjectMapper;

import java.io.IOException;
Expand All @@ -47,24 +47,24 @@ public class NestedDocuments {
private final Map<String, Weight> childObjectFilters = new HashMap<>();
private final Map<String, ObjectMapper> childObjectMappers = new HashMap<>();
private final BitSetProducer parentDocumentFilter;
private final MapperService mapperService;
private final MappingLookup mappingLookup;

/**
* Create a new NestedDocuments object for an index
* @param mapperService the index's MapperService
* @param mappingLookup the index's mapping
* @param filterProducer a function to build BitSetProducers from filter queries
*/
public NestedDocuments(MapperService mapperService, Function<Query, BitSetProducer> filterProducer) {
this.mapperService = mapperService;
if (mapperService.hasNested() == false) {
public NestedDocuments(MappingLookup mappingLookup, Function<Query, BitSetProducer> filterProducer) {
this.mappingLookup = mappingLookup;
if (mappingLookup.hasNested() == false) {
this.parentDocumentFilter = null;
} else {
this.parentDocumentFilter = filterProducer.apply(Queries.newNonNestedFilter());
for (ObjectMapper mapper : mapperService.documentMapper().getNestedParentMappers()) {
for (ObjectMapper mapper : mappingLookup.getNestedParentMappers()) {
parentObjectFilters.put(mapper.name(),
filterProducer.apply(mapper.nestedTypeFilter()));
}
for (ObjectMapper mapper : mapperService.documentMapper().getNestedMappers()) {
for (ObjectMapper mapper : mappingLookup.getNestedMappers()) {
childObjectFilters.put(mapper.name(), null);
childObjectMappers.put(mapper.name(), mapper);
}
Expand Down Expand Up @@ -98,7 +98,7 @@ private Weight getNestedChildWeight(LeafReaderContext ctx, String path) throws I
* Given an object path, returns whether or not any of its parents are plain objects
*/
public boolean hasNonNestedParent(String path) {
return mapperService.documentMapper().hasNonNestedParent(path);
return mappingLookup.hasNonNestedParent(path);
}

private class HasNestedDocuments implements LeafNestedDocuments {
Expand Down Expand Up @@ -185,7 +185,7 @@ private SearchHit.NestedIdentity loadNestedIdentity() throws IOException {
int parentNameLength;
String path = findObjectPath(doc);
while (path != null) {
String parent = mapperService.documentMapper().getNestedParent(path);
String parent = mappingLookup.getNestedParent(path);
// We have to pull a new scorer for each document here, because we advance from
// the last parent which will be behind the doc
Scorer childScorer = getNestedChildWeight(ctx, path).scorer(ctx);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -110,7 +110,7 @@ public void execute(SearchContext context) {
SearchHit[] hits = new SearchHit[context.docIdsToLoadSize()];

List<FetchSubPhaseProcessor> processors = getProcessors(context.shardTarget(), fetchContext);
NestedDocuments nestedDocuments = context.getNestedDocuments();
NestedDocuments nestedDocuments = context.getQueryShardContext().getNestedDocuments();

int currentReaderIndex = -1;
LeafReaderContext currentReaderContext = null;
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -29,7 +29,6 @@
import org.elasticsearch.index.query.ParsedQuery;
import org.elasticsearch.index.query.QueryShardContext;
import org.elasticsearch.index.shard.IndexShard;
import org.elasticsearch.search.NestedDocuments;
import org.elasticsearch.search.SearchExtBuilder;
import org.elasticsearch.search.SearchShardTarget;
import org.elasticsearch.search.aggregations.SearchContextAggregations;
Expand Down Expand Up @@ -390,11 +389,6 @@ public FetchSearchResult fetchResult() {
return in.fetchResult();
}

@Override
public NestedDocuments getNestedDocuments() {
return in.getNestedDocuments();
}

@Override
public FetchPhase fetchPhase() {
return in.fetchPhase();
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -32,7 +32,6 @@
import org.elasticsearch.index.query.ParsedQuery;
import org.elasticsearch.index.query.QueryShardContext;
import org.elasticsearch.index.shard.IndexShard;
import org.elasticsearch.search.NestedDocuments;
import org.elasticsearch.search.RescoreDocIds;
import org.elasticsearch.search.SearchExtBuilder;
import org.elasticsearch.search.SearchShardTarget;
Expand Down Expand Up @@ -311,8 +310,6 @@ public final void assignRescoreDocIds(RescoreDocIds rescoreDocIds) {

public abstract QuerySearchResult queryResult();

public abstract NestedDocuments getNestedDocuments();

public abstract FetchPhase fetchPhase();

public abstract FetchSearchResult fetchResult();
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -56,7 +56,7 @@ public void testSimpleNestedHierarchy() throws IOException {
}));

withLuceneIndex(mapperService, iw -> iw.addDocuments(doc.docs()), reader -> {
NestedDocuments nested = new NestedDocuments(mapperService, QueryBitSetProducer::new);
NestedDocuments nested = new NestedDocuments(mapperService.mappingLookup(), QueryBitSetProducer::new);
LeafNestedDocuments leaf = nested.getLeafNestedDocuments(reader.leaves().get(0));

assertNotNull(leaf.advance(0));
Expand Down Expand Up @@ -153,7 +153,7 @@ public void testMultiLevelNestedHierarchy() throws IOException {
}));

withLuceneIndex(mapperService, iw -> iw.addDocuments(doc.docs()), reader -> {
NestedDocuments nested = new NestedDocuments(mapperService, QueryBitSetProducer::new);
NestedDocuments nested = new NestedDocuments(mapperService.mappingLookup(), QueryBitSetProducer::new);
LeafNestedDocuments leaf = nested.getLeafNestedDocuments(reader.leaves().get(0));

assertNotNull(leaf.advance(0));
Expand Down Expand Up @@ -264,7 +264,7 @@ public void testNestedObjectWithinNonNestedObject() throws IOException {
}));

withLuceneIndex(mapperService, iw -> iw.addDocuments(doc.docs()), reader -> {
NestedDocuments nested = new NestedDocuments(mapperService, QueryBitSetProducer::new);
NestedDocuments nested = new NestedDocuments(mapperService.mappingLookup(), QueryBitSetProducer::new);
LeafNestedDocuments leaf = nested.getLeafNestedDocuments(reader.leaves().get(0));

assertNotNull(leaf.advance(0));
Expand Down
Loading

0 comments on commit 9d7f497

Please sign in to comment.