/* * 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. */ package org.elasticsearch.percolator; import org.apache.lucene.index.LeafReaderContext; import org.apache.lucene.index.ReaderUtil; import org.apache.lucene.search.BooleanClause; import org.apache.lucene.search.BooleanQuery; import org.apache.lucene.search.BoostQuery; import org.apache.lucene.search.ConstantScoreQuery; import org.apache.lucene.search.DisjunctionMaxQuery; import org.apache.lucene.search.IndexSearcher; import org.apache.lucene.search.Query; import org.elasticsearch.common.bytes.BytesReference; import org.elasticsearch.common.lucene.search.function.FunctionScoreQuery; import org.elasticsearch.common.settings.Settings; import org.elasticsearch.common.text.Text; import org.elasticsearch.index.query.ParsedQuery; import org.elasticsearch.search.fetch.FetchSubPhase; import org.elasticsearch.search.fetch.subphase.highlight.HighlightPhase; import org.elasticsearch.search.fetch.subphase.highlight.Highlighter; import org.elasticsearch.search.fetch.subphase.highlight.SearchContextHighlight; import org.elasticsearch.search.SearchHit; import org.elasticsearch.search.internal.SearchContext; import org.elasticsearch.search.internal.SubSearchContext; import java.io.IOException; import java.util.Collections; import java.util.List; import java.util.Map; /** * Highlighting in the case of the percolate query is a bit different, because the PercolateQuery itself doesn't get highlighted, * but the source of the PercolateQuery gets highlighted by each hit containing a query. */ public final class PercolatorHighlightSubFetchPhase extends HighlightPhase { public PercolatorHighlightSubFetchPhase(Settings settings, Map<String, Highlighter> highlighters) { super(settings, highlighters); } boolean hitsExecutionNeeded(SearchContext context) { // for testing return context.highlight() != null && locatePercolatorQuery(context.query()) != null; } @Override public void hitsExecute(SearchContext context, SearchHit[] hits) { if (hitsExecutionNeeded(context) == false) { return; } PercolateQuery percolateQuery = locatePercolatorQuery(context.query()); if (percolateQuery == null) { // shouldn't happen as we checked for the existence of a percolator query in hitsExecutionNeeded(...) throw new IllegalStateException("couldn't locate percolator query"); } List<LeafReaderContext> ctxs = context.searcher().getIndexReader().leaves(); IndexSearcher percolatorIndexSearcher = percolateQuery.getPercolatorIndexSearcher(); PercolateQuery.QueryStore queryStore = percolateQuery.getQueryStore(); LeafReaderContext percolatorLeafReaderContext = percolatorIndexSearcher.getIndexReader().leaves().get(0); FetchSubPhase.HitContext hitContext = new FetchSubPhase.HitContext(); SubSearchContext subSearchContext = createSubSearchContext(context, percolatorLeafReaderContext, percolateQuery.getDocumentSource()); for (SearchHit hit : hits) { final Query query; try { LeafReaderContext ctx = ctxs.get(ReaderUtil.subIndex(hit.docId(), ctxs)); int segmentDocId = hit.docId() - ctx.docBase; query = queryStore.getQueries(ctx).apply(segmentDocId); } catch (IOException e) { throw new RuntimeException(e); } if (query != null) { subSearchContext.parsedQuery(new ParsedQuery(query)); hitContext.reset( new SearchHit(0, "unknown", new Text(percolateQuery.getDocumentType()), Collections.emptyMap()), percolatorLeafReaderContext, 0, percolatorIndexSearcher ); hitContext.cache().clear(); super.hitExecute(subSearchContext, hitContext); hit.getHighlightFields().putAll(hitContext.hit().getHighlightFields()); } } } static PercolateQuery locatePercolatorQuery(Query query) { if (query instanceof PercolateQuery) { return (PercolateQuery) query; } else if (query instanceof BooleanQuery) { for (BooleanClause clause : ((BooleanQuery) query).clauses()) { PercolateQuery result = locatePercolatorQuery(clause.getQuery()); if (result != null) { return result; } } } else if (query instanceof DisjunctionMaxQuery) { for (Query disjunct : ((DisjunctionMaxQuery) query).getDisjuncts()) { PercolateQuery result = locatePercolatorQuery(disjunct); if (result != null) { return result; } } } else if (query instanceof ConstantScoreQuery) { return locatePercolatorQuery(((ConstantScoreQuery) query).getQuery()); } else if (query instanceof BoostQuery) { return locatePercolatorQuery(((BoostQuery) query).getQuery()); } else if (query instanceof FunctionScoreQuery) { return locatePercolatorQuery(((FunctionScoreQuery) query).getSubQuery()); } return null; } private SubSearchContext createSubSearchContext(SearchContext context, LeafReaderContext leafReaderContext, BytesReference source) { SubSearchContext subSearchContext = new SubSearchContext(context); subSearchContext.highlight(new SearchContextHighlight(context.highlight().fields())); // Enforce highlighting by source, because MemoryIndex doesn't support stored fields. subSearchContext.highlight().globalForceSource(true); subSearchContext.lookup().source().setSegmentAndDocument(leafReaderContext, 0); subSearchContext.lookup().source().setSource(source); return subSearchContext; } }