Ensure either success or failure path for SearchOperationListener is called (#37467)

Today we have several implementations of executing SearchOperationListener
in SearchService. While all of them seem to be safe at least on, the one that
executes scroll searches can cause illegal execution of SearchOperationListener
that can then in-turn trigger assertions in ShardSearchStats. This change
adds a SearchOperationListenerExecutor that uses try-with blocks to ensure
listeners are called in a safe way.

Relates to #37185
This commit is contained in:
Simon Willnauer 2019-01-23 12:38:44 +01:00 committed by GitHub
parent 100537fbc3
commit 4ec3a6d922
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
1 changed files with 95 additions and 76 deletions

View File

@ -24,7 +24,6 @@ import org.apache.logging.log4j.Logger;
import org.apache.lucene.search.FieldDoc; import org.apache.lucene.search.FieldDoc;
import org.apache.lucene.search.TopDocs; import org.apache.lucene.search.TopDocs;
import org.elasticsearch.ElasticsearchException; import org.elasticsearch.ElasticsearchException;
import org.elasticsearch.ExceptionsHelper;
import org.elasticsearch.action.ActionListener; import org.elasticsearch.action.ActionListener;
import org.elasticsearch.action.OriginalIndices; import org.elasticsearch.action.OriginalIndices;
import org.elasticsearch.action.search.SearchTask; import org.elasticsearch.action.search.SearchTask;
@ -329,7 +328,7 @@ public class SearchService extends AbstractLifecycleComponent implements IndexEv
} catch (Exception e) { } catch (Exception e) {
logger.trace("Dfs phase failed", e); logger.trace("Dfs phase failed", e);
processFailure(context, e); processFailure(context, e);
throw ExceptionsHelper.convertToRuntime(e); throw e;
} finally { } finally {
cleanContext(context); cleanContext(context);
} }
@ -380,29 +379,24 @@ public class SearchService extends AbstractLifecycleComponent implements IndexEv
}); });
} }
private SearchPhaseResult executeQueryPhase(ShardSearchRequest request, SearchTask task) throws IOException { private SearchPhaseResult executeQueryPhase(ShardSearchRequest request, SearchTask task) throws Exception {
final SearchContext context = createAndPutContext(request); final SearchContext context = createAndPutContext(request);
final SearchOperationListener operationListener = context.indexShard().getSearchOperationListener();
context.incRef(); context.incRef();
boolean queryPhaseSuccess = false;
try { try {
context.setTask(task); context.setTask(task);
operationListener.onPreQueryPhase(context); final long afterQueryTime;
long time = System.nanoTime(); try (SearchOperationListenerExecutor executor = new SearchOperationListenerExecutor(context)) {
contextProcessing(context); contextProcessing(context);
loadOrExecuteQueryPhase(request, context); loadOrExecuteQueryPhase(request, context);
if (context.queryResult().hasSearchContext() == false && context.scrollContext() == null) { if (context.queryResult().hasSearchContext() == false && context.scrollContext() == null) {
freeContext(context.id()); freeContext(context.id());
} else { } else {
contextProcessedSuccessfully(context); contextProcessedSuccessfully(context);
} }
final long afterQueryTime = System.nanoTime(); afterQueryTime = executor.success();
queryPhaseSuccess = true; }
operationListener.onQueryPhase(context, afterQueryTime - time);
if (request.numberOfShards() == 1) { if (request.numberOfShards() == 1) {
return executeFetchPhase(context, operationListener, afterQueryTime); return executeFetchPhase(context, afterQueryTime);
} }
return context.queryResult(); return context.queryResult();
} catch (Exception e) { } catch (Exception e) {
@ -411,21 +405,16 @@ public class SearchService extends AbstractLifecycleComponent implements IndexEv
e = (e.getCause() == null || e.getCause() instanceof Exception) ? e = (e.getCause() == null || e.getCause() instanceof Exception) ?
(Exception) e.getCause() : new ElasticsearchException(e.getCause()); (Exception) e.getCause() : new ElasticsearchException(e.getCause());
} }
if (!queryPhaseSuccess) {
operationListener.onFailedQueryPhase(context);
}
logger.trace("Query phase failed", e); logger.trace("Query phase failed", e);
processFailure(context, e); processFailure(context, e);
throw ExceptionsHelper.convertToRuntime(e); throw e;
} finally { } finally {
cleanContext(context); cleanContext(context);
} }
} }
private QueryFetchSearchResult executeFetchPhase(SearchContext context, SearchOperationListener operationListener, private QueryFetchSearchResult executeFetchPhase(SearchContext context, long afterQueryTime) {
long afterQueryTime) { try (SearchOperationListenerExecutor executor = new SearchOperationListenerExecutor(context, true, afterQueryTime)){
operationListener.onPreFetchPhase(context);
try {
shortcutDocIdsToLoad(context); shortcutDocIdsToLoad(context);
fetchPhase.execute(context); fetchPhase.execute(context);
if (fetchPhaseShouldFreeContext(context)) { if (fetchPhaseShouldFreeContext(context)) {
@ -433,34 +422,27 @@ public class SearchService extends AbstractLifecycleComponent implements IndexEv
} else { } else {
contextProcessedSuccessfully(context); contextProcessedSuccessfully(context);
} }
} catch (Exception e) { executor.success();
operationListener.onFailedFetchPhase(context);
throw ExceptionsHelper.convertToRuntime(e);
} }
operationListener.onFetchPhase(context, System.nanoTime() - afterQueryTime);
return new QueryFetchSearchResult(context.queryResult(), context.fetchResult()); return new QueryFetchSearchResult(context.queryResult(), context.fetchResult());
} }
public void executeQueryPhase(InternalScrollSearchRequest request, SearchTask task, ActionListener<ScrollQuerySearchResult> listener) { public void executeQueryPhase(InternalScrollSearchRequest request, SearchTask task, ActionListener<ScrollQuerySearchResult> listener) {
runAsync(request.id(), () -> { runAsync(request.id(), () -> {
final SearchContext context = findContext(request.id(), request); final SearchContext context = findContext(request.id(), request);
SearchOperationListener operationListener = context.indexShard().getSearchOperationListener();
context.incRef(); context.incRef();
try { try (SearchOperationListenerExecutor executor = new SearchOperationListenerExecutor(context)) {
context.setTask(task); context.setTask(task);
operationListener.onPreQueryPhase(context);
long time = System.nanoTime();
contextProcessing(context); contextProcessing(context);
processScroll(request, context); processScroll(request, context);
queryPhase.execute(context); queryPhase.execute(context);
contextProcessedSuccessfully(context); contextProcessedSuccessfully(context);
operationListener.onQueryPhase(context, System.nanoTime() - time); executor.success();
return new ScrollQuerySearchResult(context.queryResult(), context.shardTarget()); return new ScrollQuerySearchResult(context.queryResult(), context.shardTarget());
} catch (Exception e) { } catch (Exception e) {
operationListener.onFailedQueryPhase(context);
logger.trace("Query phase failed", e); logger.trace("Query phase failed", e);
processFailure(context, e); processFailure(context, e);
throw ExceptionsHelper.convertToRuntime(e); throw e;
} finally { } finally {
cleanContext(context); cleanContext(context);
} }
@ -471,15 +453,10 @@ public class SearchService extends AbstractLifecycleComponent implements IndexEv
runAsync(request.id(), () -> { runAsync(request.id(), () -> {
final SearchContext context = findContext(request.id(), request); final SearchContext context = findContext(request.id(), request);
context.setTask(task); context.setTask(task);
IndexShard indexShard = context.indexShard();
SearchOperationListener operationListener = indexShard.getSearchOperationListener();
context.incRef(); context.incRef();
try { try (SearchOperationListenerExecutor executor = new SearchOperationListenerExecutor(context)) {
contextProcessing(context); contextProcessing(context);
context.searcher().setAggregatedDfs(request.dfs()); context.searcher().setAggregatedDfs(request.dfs());
operationListener.onPreQueryPhase(context);
long time = System.nanoTime();
queryPhase.execute(context); queryPhase.execute(context);
if (context.queryResult().hasSearchContext() == false && context.scrollContext() == null) { if (context.queryResult().hasSearchContext() == false && context.scrollContext() == null) {
// no hits, we can release the context since there will be no fetch phase // no hits, we can release the context since there will be no fetch phase
@ -487,13 +464,12 @@ public class SearchService extends AbstractLifecycleComponent implements IndexEv
} else { } else {
contextProcessedSuccessfully(context); contextProcessedSuccessfully(context);
} }
operationListener.onQueryPhase(context, System.nanoTime() - time); executor.success();
return context.queryResult(); return context.queryResult();
} catch (Exception e) { } catch (Exception e) {
operationListener.onFailedQueryPhase(context);
logger.trace("Query phase failed", e); logger.trace("Query phase failed", e);
processFailure(context, e); processFailure(context, e);
throw ExceptionsHelper.convertToRuntime(e); throw e;
} finally { } finally {
cleanContext(context); cleanContext(context);
} }
@ -527,28 +503,19 @@ public class SearchService extends AbstractLifecycleComponent implements IndexEv
ActionListener<ScrollQueryFetchSearchResult> listener) { ActionListener<ScrollQueryFetchSearchResult> listener) {
runAsync(request.id(), () -> { runAsync(request.id(), () -> {
final SearchContext context = findContext(request.id(), request); final SearchContext context = findContext(request.id(), request);
context.incRef();
try {
context.setTask(task); context.setTask(task);
context.incRef();
try (SearchOperationListenerExecutor executor = new SearchOperationListenerExecutor(context)){
contextProcessing(context); contextProcessing(context);
SearchOperationListener operationListener = context.indexShard().getSearchOperationListener();
processScroll(request, context); processScroll(request, context);
operationListener.onPreQueryPhase(context);
final long time = System.nanoTime();
try {
queryPhase.execute(context); queryPhase.execute(context);
} catch (Exception e) { final long afterQueryTime = executor.success();
operationListener.onFailedQueryPhase(context); QueryFetchSearchResult fetchSearchResult = executeFetchPhase(context, afterQueryTime);
throw ExceptionsHelper.convertToRuntime(e);
}
long afterQueryTime = System.nanoTime();
operationListener.onQueryPhase(context, afterQueryTime - time);
QueryFetchSearchResult fetchSearchResult = executeFetchPhase(context, operationListener, afterQueryTime);
return new ScrollQueryFetchSearchResult(fetchSearchResult, context.shardTarget()); return new ScrollQueryFetchSearchResult(fetchSearchResult, context.shardTarget());
} catch (Exception e) { } catch (Exception e) {
logger.trace("Fetch phase failed", e); logger.trace("Fetch phase failed", e);
processFailure(context, e); processFailure(context, e);
throw ExceptionsHelper.convertToRuntime(e); throw e;
} finally { } finally {
cleanContext(context); cleanContext(context);
} }
@ -558,7 +525,6 @@ public class SearchService extends AbstractLifecycleComponent implements IndexEv
public void executeFetchPhase(ShardFetchRequest request, SearchTask task, ActionListener<FetchSearchResult> listener) { public void executeFetchPhase(ShardFetchRequest request, SearchTask task, ActionListener<FetchSearchResult> listener) {
runAsync(request.id(), () -> { runAsync(request.id(), () -> {
final SearchContext context = findContext(request.id(), request); final SearchContext context = findContext(request.id(), request);
final SearchOperationListener operationListener = context.indexShard().getSearchOperationListener();
context.incRef(); context.incRef();
try { try {
context.setTask(task); context.setTask(task);
@ -567,21 +533,20 @@ public class SearchService extends AbstractLifecycleComponent implements IndexEv
context.scrollContext().lastEmittedDoc = request.lastEmittedDoc(); context.scrollContext().lastEmittedDoc = request.lastEmittedDoc();
} }
context.docIdsToLoad(request.docIds(), 0, request.docIdsSize()); context.docIdsToLoad(request.docIds(), 0, request.docIdsSize());
operationListener.onPreFetchPhase(context); try (SearchOperationListenerExecutor executor = new SearchOperationListenerExecutor(context, true, System.nanoTime())) {
long time = System.nanoTime();
fetchPhase.execute(context); fetchPhase.execute(context);
if (fetchPhaseShouldFreeContext(context)) { if (fetchPhaseShouldFreeContext(context)) {
freeContext(request.id()); freeContext(request.id());
} else { } else {
contextProcessedSuccessfully(context); contextProcessedSuccessfully(context);
} }
operationListener.onFetchPhase(context, System.nanoTime() - time); executor.success();
}
return context.fetchResult(); return context.fetchResult();
} catch (Exception e) { } catch (Exception e) {
operationListener.onFailedFetchPhase(context);
logger.trace("Fetch phase failed", e); logger.trace("Fetch phase failed", e);
processFailure(context, e); processFailure(context, e);
throw ExceptionsHelper.convertToRuntime(e); throw e;
} finally { } finally {
cleanContext(context); cleanContext(context);
} }
@ -661,7 +626,7 @@ public class SearchService extends AbstractLifecycleComponent implements IndexEv
context.lowLevelCancellation(lowLevelCancellation); context.lowLevelCancellation(lowLevelCancellation);
} catch (Exception e) { } catch (Exception e) {
context.close(); context.close();
throw ExceptionsHelper.convertToRuntime(e); throw e;
} }
return context; return context;
@ -733,7 +698,7 @@ public class SearchService extends AbstractLifecycleComponent implements IndexEv
} }
} }
private void contextScrollKeepAlive(SearchContext context, long keepAlive) throws IOException { private void contextScrollKeepAlive(SearchContext context, long keepAlive) {
if (keepAlive > maxKeepAlive) { if (keepAlive > maxKeepAlive) {
throw new IllegalArgumentException( throw new IllegalArgumentException(
"Keep alive for scroll (" + TimeValue.timeValueMillis(keepAlive) + ") is too large. " + "Keep alive for scroll (" + TimeValue.timeValueMillis(keepAlive) + ") is too large. " +
@ -991,7 +956,7 @@ public class SearchService extends AbstractLifecycleComponent implements IndexEv
context.docIdsToLoad(docIdsToLoad, 0, docIdsToLoad.length); context.docIdsToLoad(docIdsToLoad, 0, docIdsToLoad.length);
} }
private void processScroll(InternalScrollSearchRequest request, SearchContext context) throws IOException { private void processScroll(InternalScrollSearchRequest request, SearchContext context) {
// process scroll // process scroll
context.from(context.from() + context.size()); context.from(context.from() + context.size());
context.scrollContext().scroll = request.scroll(); context.scrollContext().scroll = request.scroll();
@ -1147,4 +1112,58 @@ public class SearchService extends AbstractLifecycleComponent implements IndexEv
return canMatch; return canMatch;
} }
} }
/**
* This helper class ensures we only execute either the success or the failure path for {@link SearchOperationListener}.
* This is crucial for some implementations like {@link org.elasticsearch.index.search.stats.ShardSearchStats}.
*/
private static final class SearchOperationListenerExecutor implements AutoCloseable {
private final SearchOperationListener listener;
private final SearchContext context;
private final long time;
private final boolean fetch;
private long afterQueryTime = -1;
private boolean closed = false;
SearchOperationListenerExecutor(SearchContext context) {
this(context, false, System.nanoTime());
}
SearchOperationListenerExecutor(SearchContext context, boolean fetch, long startTime) {
this.listener = context.indexShard().getSearchOperationListener();
this.context = context;
time = startTime;
this.fetch = fetch;
if (fetch) {
listener.onPreFetchPhase(context);
} else {
listener.onPreQueryPhase(context);
}
}
long success() {
return afterQueryTime = System.nanoTime();
}
@Override
public void close() {
assert closed == false : "already closed - while technically ok double closing is a likely a bug in this case";
if (closed == false) {
closed = true;
if (afterQueryTime != -1) {
if (fetch) {
listener.onFetchPhase(context, afterQueryTime - time);
} else {
listener.onQueryPhase(context, afterQueryTime - time);
}
} else {
if (fetch) {
listener.onFailedFetchPhase(context);
} else {
listener.onFailedQueryPhase(context);
}
}
}
}
}
} }