Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Inject the backend query type map for pluggable backend selection #21121

Merged
merged 5 commits into from
Dec 6, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -24,8 +24,6 @@
import com.google.inject.multibindings.MapBinder;
import org.graylog.plugins.views.ViewsModule;
import org.graylog.plugins.views.search.SearchType;
import org.graylog.plugins.views.search.engine.GeneratedQueryContext;
import org.graylog.plugins.views.search.engine.QueryBackend;
import org.graylog.plugins.views.search.export.ExportBackend;
import org.graylog.plugins.views.search.searchtypes.MessageList;
import org.graylog.plugins.views.search.searchtypes.events.EventList;
Expand Down Expand Up @@ -86,8 +84,7 @@ public ViewsESBackendModule(SearchVersion supportedSearchVersion) {
protected void configure() {
install(new FactoryModuleBuilder().build(ESGeneratedQueryContext.Factory.class));

bindForVersion(supportedSearchVersion, new TypeLiteral<QueryBackend<? extends GeneratedQueryContext>>() {})
.to(ElasticsearchBackend.class);
registerVersionedQueryBackend(supportedSearchVersion, ElasticsearchBackend.class);

registerESSearchTypeHandler(MessageList.NAME, ESMessageList.class);
registerESSearchTypeHandler(EventList.NAME, ESEventList.class);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -25,8 +25,6 @@
import com.google.inject.multibindings.OptionalBinder;
import org.graylog.plugins.views.ViewsModule;
import org.graylog.plugins.views.search.SearchType;
import org.graylog.plugins.views.search.engine.GeneratedQueryContext;
import org.graylog.plugins.views.search.engine.QueryBackend;
import org.graylog.plugins.views.search.export.ExportBackend;
import org.graylog.plugins.views.search.searchtypes.MessageList;
import org.graylog.plugins.views.search.searchtypes.events.EventList;
Expand Down Expand Up @@ -89,8 +87,7 @@ public ViewsOSBackendModule(SearchVersion supportedSearchVersion) {
protected void configure() {
install(new FactoryModuleBuilder().build(OSGeneratedQueryContext.Factory.class));

bindForVersion(supportedSearchVersion, new TypeLiteral<QueryBackend<? extends GeneratedQueryContext>>() {})
.to(OpenSearchBackend.class);
registerVersionedQueryBackend(supportedSearchVersion, OpenSearchBackend.class);

registerOSSearchTypeHandler(MessageList.NAME, OSMessageList.class);
registerOSSearchTypeHandler(EventList.NAME, OSEventListDelegate.class);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -181,6 +181,8 @@ protected void configure() {
registerJacksonSubtype(StreamCategoryFilter.class);
registerJacksonSubtype(QueryStringFilter.class);

// to support unversioned query backends
queryBackendBinder();
// query backends for jackson
registerJacksonSubtype(ElasticsearchQueryString.class);

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -18,7 +18,6 @@

import com.google.inject.TypeLiteral;
import com.google.inject.binder.LinkedBindingBuilder;
import com.google.inject.binder.ScopedBindingBuilder;
import com.google.inject.multibindings.MapBinder;
import com.google.inject.multibindings.Multibinder;
import com.google.inject.multibindings.OptionalBinder;
Expand Down Expand Up @@ -85,15 +84,15 @@ protected void registerPivotAggregationFunction(String name, String description,
seriesSpecBinder().addBinding(name).toInstance(SeriesDescription.create(name, description));
}

protected MapBinder<String, QueryBackend<? extends GeneratedQueryContext>> queryBackendBinder(SearchVersion version) {
protected MapBinder<String, QueryBackend<? extends GeneratedQueryContext>> queryBackendBinder() {
return MapBinder.newMapBinder(binder(),
TypeLiteral.get(String.class),
new TypeLiteral<QueryBackend<? extends GeneratedQueryContext>>() {});
new TypeLiteral<>() {});

}

protected ScopedBindingBuilder registerQueryBackend(SearchVersion version, String name, Class<? extends QueryBackend<? extends GeneratedQueryContext>> implementation) {
return queryBackendBinder(version).addBinding(name).to(implementation);
protected void registerVersionedQueryBackend(SearchVersion version, Class<? extends QueryBackend<? extends GeneratedQueryContext>> implementation) {
bindForVersion(version, new TypeLiteral<QueryBackend<? extends GeneratedQueryContext>>() {}).to(implementation);
}

protected void registerESQueryDecorator(Class<? extends QueryStringDecorator> esQueryDecorator) {
Expand Down Expand Up @@ -135,4 +134,6 @@ protected void registerSearchValidator(Class<? extends SearchValidator> validato
protected Multibinder<SearchValidator> searchValidatorBinder() {
return Multibinder.newSetBinder(binder(), SearchValidator.class);
}


}
Original file line number Diff line number Diff line change
Expand Up @@ -27,9 +27,11 @@
import org.graylog.plugins.views.search.QueryResult;
import org.graylog.plugins.views.search.Search;
import org.graylog.plugins.views.search.SearchJob;
import org.graylog.plugins.views.search.elasticsearch.ElasticsearchQueryString;
import org.graylog.plugins.views.search.errors.QueryError;
import org.graylog.plugins.views.search.errors.SearchError;
import org.graylog.plugins.views.search.errors.SearchException;
import org.graylog2.storage.providers.ElasticsearchBackendProvider;
import org.joda.time.DateTimeZone;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
Expand All @@ -52,13 +54,16 @@ public class QueryEngine {

// TODO proper thread pool with tunable settings
private final Executor queryPool = Executors.newFixedThreadPool(4, new ThreadFactoryBuilder().setNameFormat("query-engine-%d").build());
private final QueryBackend<? extends GeneratedQueryContext> backend;
private final ElasticsearchBackendProvider elasticsearchBackendProvider;
private final Map<String, QueryBackend<? extends GeneratedQueryContext>> unversionedBackends;

@Inject
public QueryEngine(QueryBackend<? extends GeneratedQueryContext> backend,
public QueryEngine(ElasticsearchBackendProvider elasticsearchBackendProvider,
Map<String, QueryBackend<? extends GeneratedQueryContext>> unversionedBackends,
Set<QueryMetadataDecorator> queryMetadataDecorators,
QueryParser queryParser) {
this.backend = backend;
this.elasticsearchBackendProvider = elasticsearchBackendProvider;
this.unversionedBackends = unversionedBackends;
this.queryMetadataDecorators = queryMetadataDecorators;
this.queryParser = queryParser;
}
Expand All @@ -75,6 +80,7 @@ public QueryMetadata parse(Search search, Query query) {
public ExplainResults explain(SearchJob searchJob, Set<SearchError> validationErrors, DateTimeZone timezone) {
final Map<String, ExplainResults.QueryExplainResult> queries = searchJob.getSearch().queries().stream()
.collect(Collectors.toMap(Query::id, q -> {
var backend = getBackendForQuery(q);
final GeneratedQueryContext generatedQueryContext = backend.generate(q, Set.of(), timezone);

return backend.explain(searchJob, q, generatedQueryContext);
Expand Down Expand Up @@ -116,6 +122,7 @@ public SearchJob execute(SearchJob searchJob, Set<SearchError> validationErrors,
}

private QueryResult prepareAndRun(SearchJob searchJob, Query query, Set<SearchError> validationErrors, DateTimeZone timezone) {
final var backend = getBackendForQuery(query);
LOG.debug("[{}] Using {} to generate query", query.id(), backend);
// with all the results done, we can execute the current query and eventually complete our own result
// if any of this throws an exception, the handle in #execute will convert it to an error and return a "failed" result instead
Expand All @@ -137,4 +144,15 @@ private boolean isQueryWithError(Collection<SearchError> validationErrors, Query
.map(QueryError::queryId)
.anyMatch(id -> Objects.equals(id, query.id()));
}

private QueryBackend<? extends GeneratedQueryContext> getBackendForQuery(Query query) {
var backendQuery = query.query();
if (backendQuery.type().equals(ElasticsearchQueryString.NAME)) {
return elasticsearchBackendProvider.get();
}
if (unversionedBackends.containsKey(backendQuery.type())) {
return unversionedBackends.get(backendQuery.type());
}
throw new IllegalArgumentException("Unknown backend type: " + backendQuery.type());
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -38,6 +38,10 @@ public DecorateQueryStringsNormalizer(QueryStringDecorators queryStringDecorator

@Override
public Query normalizeQuery(final Query query, final ParameterProvider parameterProvider) {
// this only makes sense for ElasticsearchQueryString instances, don't touch any other type
if (!(query.query() instanceof ElasticsearchQueryString)) {
return query;
}
return query.toBuilder()
.query(ElasticsearchQueryString.of(this.queryStringDecorators.decorate(query.query().queryString(), parameterProvider, query)))
.filter(normalizeFilter(query.filter(), query, parameterProvider))
Expand Down
Loading