SQL: GROUP BY use composite instead of terms aggregation (elastic/x-pack-elasticsearch#4282)
Rewrote the GROUP BY to use composite aggregation instead of terms (and everything that comes with it) but instead rely on composite aggregation This not only works better but simplifies the code complexity since composite is a straight, two-level tree: 1. root/group-by/composite-keys 2. (metric) aggregations This removes a lot of complexity from all stages that involve creating, assembling and especially parsing the results. By moving to composite agg, the aggregation/GROUP BY are now pageable so the consumer/listener had to be extended to include a dedicated cursor and specific (bucket) extractors inline with the scroll requests. While at it, also improved the support for implicit GROUP BY by formalizing it (previously it supported only counts and no other agg). In addition: Fixed a JDBC bug that caused incorrect timeout to be passed Improved the returned RowSet a bit and add better naming Pick up @Nullable move from core Make sure to specify the TimeZone for DateTimeHistogram extraction Add missing javadoc To avoid delegating NamedWriteableRegistry (NWR) and to keep the scope clean, SQL writeables now handle their own serialization, keeping the boundary between the Elasticsearch's NWR in place. Pass NamedWriteableRegistry only when looking at the next page To keep in line with the existing patter and simplify the code bureaucracy, the deserialization happens directly. Since the SearchSourceBuilder deserialization happens explicitly (and it's otherwise opaque), the declarative invocation isn't necessary anymore. Add a bit more randomization in tests Original commit: elastic/x-pack-elasticsearch@f5af046386
This commit is contained in:
parent
df61103b05
commit
e5d18ad45c
|
@ -29,11 +29,12 @@ class JdbcStatement implements Statement, JdbcWrapper {
|
|||
private boolean ignoreResultSetClose = false;
|
||||
|
||||
protected JdbcResultSet rs;
|
||||
final RequestMeta requestMeta = new RequestMeta();
|
||||
final RequestMeta requestMeta;
|
||||
|
||||
JdbcStatement(JdbcConnection jdbcConnection, JdbcConfiguration info) {
|
||||
this.con = jdbcConnection;
|
||||
this.cfg = info;
|
||||
this.requestMeta = new RequestMeta(info.pageSize(), info.pageTimeout(), info.queryTimeout());
|
||||
}
|
||||
|
||||
@Override
|
||||
|
@ -111,7 +112,7 @@ class JdbcStatement implements Statement, JdbcWrapper {
|
|||
@Override
|
||||
public int getQueryTimeout() throws SQLException {
|
||||
checkOpen();
|
||||
return (int) TimeUnit.MILLISECONDS.toSeconds(requestMeta.timeoutInMs());
|
||||
return (int) TimeUnit.MILLISECONDS.toSeconds(requestMeta.queryTimeoutInMs());
|
||||
}
|
||||
|
||||
@Override
|
||||
|
@ -120,7 +121,7 @@ class JdbcStatement implements Statement, JdbcWrapper {
|
|||
if (seconds < 0) {
|
||||
throw new SQLException("Query timeout must be positive");
|
||||
}
|
||||
requestMeta.timeout(TimeUnit.SECONDS.toMillis(seconds));
|
||||
requestMeta.queryTimeout(TimeUnit.SECONDS.toMillis(seconds));
|
||||
}
|
||||
|
||||
@Override
|
||||
|
@ -215,14 +216,13 @@ class JdbcStatement implements Statement, JdbcWrapper {
|
|||
@Override
|
||||
public int getFetchSize() throws SQLException {
|
||||
checkOpen();
|
||||
int fetchSize = requestMeta.fetchSize();
|
||||
// the spec is somewhat unclear. It looks like there are 3 states:
|
||||
// unset (in this case -1 which the user cannot set) - in this case, the default fetch size is returned
|
||||
// 0 meaning the hint is disabled (the user has called setFetch)
|
||||
// >0 means actual hint
|
||||
|
||||
// tl;dr - if invalid, it means it was not set so return default - otherwise return the set value
|
||||
return fetchSize < 0 ? cfg.pageSize() : fetchSize;
|
||||
// tl;dr - unless the user set it, returning the default is fine
|
||||
return requestMeta.fetchSize();
|
||||
}
|
||||
|
||||
@Override
|
||||
|
|
|
@ -45,7 +45,7 @@ public class JdbcHttpClient {
|
|||
public Cursor query(String sql, List<SqlTypedParamValue> params, RequestMeta meta) throws SQLException {
|
||||
int fetch = meta.fetchSize() > 0 ? meta.fetchSize() : conCfg.pageSize();
|
||||
SqlQueryRequest sqlRequest = new SqlQueryRequest(AbstractSqlRequest.Mode.JDBC, sql, params, null, DateTimeZone.UTC, fetch,
|
||||
TimeValue.timeValueMillis(meta.timeoutInMs()), TimeValue.timeValueMillis(meta.timeoutInMs()), "");
|
||||
TimeValue.timeValueMillis(meta.timeoutInMs()), TimeValue.timeValueMillis(meta.queryTimeoutInMs()), "");
|
||||
SqlQueryResponse response = httpClient.query(sqlRequest);
|
||||
return new DefaultCursor(this, response.cursor(), toJdbcColumnInfo(response.columns()), response.rows(), meta);
|
||||
}
|
||||
|
@ -55,11 +55,10 @@ public class JdbcHttpClient {
|
|||
* the scroll id to use to fetch the next page.
|
||||
*/
|
||||
public Tuple<String, List<List<Object>>> nextPage(String cursor, RequestMeta meta) throws SQLException {
|
||||
TimeValue timeValue = TimeValue.timeValueMillis(meta.timeoutInMs());
|
||||
SqlQueryRequest sqlRequest = new SqlQueryRequest().cursor(cursor);
|
||||
sqlRequest.mode(AbstractSqlRequest.Mode.JDBC);
|
||||
sqlRequest.requestTimeout(timeValue);
|
||||
sqlRequest.pageTimeout(timeValue);
|
||||
sqlRequest.requestTimeout(TimeValue.timeValueMillis(meta.timeoutInMs()));
|
||||
sqlRequest.pageTimeout(TimeValue.timeValueMillis(meta.queryTimeoutInMs()));
|
||||
SqlQueryResponse response = httpClient.query(sqlRequest);
|
||||
return new Tuple<>(response.cursor(), response.rows());
|
||||
}
|
||||
|
|
|
@ -9,14 +9,17 @@ public class RequestMeta {
|
|||
|
||||
private int fetchSize;
|
||||
private long timeoutInMs;
|
||||
private long queryTimeoutInMs;
|
||||
|
||||
public RequestMeta() {
|
||||
this(-1, 0);
|
||||
}
|
||||
|
||||
public RequestMeta(int fetchSize, int timeout) {
|
||||
public RequestMeta(int fetchSize, long timeout, long queryTimeoutInMs) {
|
||||
this.fetchSize = fetchSize;
|
||||
this.timeoutInMs = timeout;
|
||||
this.queryTimeoutInMs = queryTimeoutInMs;
|
||||
}
|
||||
|
||||
public RequestMeta queryTimeout(long timeout) {
|
||||
this.queryTimeoutInMs = timeout;
|
||||
return this;
|
||||
}
|
||||
|
||||
public RequestMeta timeout(long timeout) {
|
||||
|
@ -37,4 +40,7 @@ public class RequestMeta {
|
|||
return timeoutInMs;
|
||||
}
|
||||
|
||||
public long queryTimeoutInMs() {
|
||||
return queryTimeoutInMs;
|
||||
}
|
||||
}
|
||||
|
|
|
@ -7,7 +7,7 @@ package org.elasticsearch.xpack.sql;
|
|||
|
||||
import org.elasticsearch.ElasticsearchException;
|
||||
|
||||
public class SqlException extends ElasticsearchException {
|
||||
public abstract class SqlException extends ElasticsearchException {
|
||||
public SqlException(String message, Throwable cause, boolean enableSuppression, boolean writableStackTrace) {
|
||||
super(message, cause, enableSuppression, writableStackTrace);
|
||||
}
|
||||
|
|
|
@ -5,13 +5,13 @@
|
|||
*/
|
||||
package org.elasticsearch.xpack.sql.analysis.analyzer;
|
||||
|
||||
import org.elasticsearch.common.util.Comparators;
|
||||
import org.elasticsearch.xpack.sql.analysis.AnalysisException;
|
||||
import org.elasticsearch.xpack.sql.analysis.analyzer.Verifier.Failure;
|
||||
import org.elasticsearch.xpack.sql.analysis.index.IndexResolution;
|
||||
import org.elasticsearch.xpack.sql.capabilities.Resolvables;
|
||||
import org.elasticsearch.xpack.sql.expression.Alias;
|
||||
import org.elasticsearch.xpack.sql.expression.Attribute;
|
||||
import org.elasticsearch.xpack.sql.expression.AttributeMap;
|
||||
import org.elasticsearch.xpack.sql.expression.AttributeSet;
|
||||
import org.elasticsearch.xpack.sql.expression.Expression;
|
||||
import org.elasticsearch.xpack.sql.expression.Expressions;
|
||||
|
@ -55,7 +55,6 @@ import org.joda.time.DateTimeZone;
|
|||
import java.util.ArrayList;
|
||||
import java.util.Arrays;
|
||||
import java.util.Collection;
|
||||
import java.util.Comparator;
|
||||
import java.util.LinkedHashMap;
|
||||
import java.util.LinkedHashSet;
|
||||
import java.util.List;
|
||||
|
@ -110,7 +109,7 @@ public class Analyzer extends RuleExecutor<LogicalPlan> {
|
|||
new ResolveFunctions(),
|
||||
new ResolveAliases(),
|
||||
new ProjectedAggregations(),
|
||||
new ResolveAggsInHavingAndOrderBy()
|
||||
new ResolveAggsInHaving()
|
||||
//new ImplicitCasting()
|
||||
);
|
||||
// TODO: this might be removed since the deduplication happens already in ResolveFunctions
|
||||
|
@ -159,7 +158,7 @@ public class Analyzer extends RuleExecutor<LogicalPlan> {
|
|||
// Shared methods around the analyzer rules
|
||||
//
|
||||
|
||||
private static Attribute resolveAgainstList(UnresolvedAttribute u, List<Attribute> attrList) {
|
||||
private static Attribute resolveAgainstList(UnresolvedAttribute u, Collection<Attribute> attrList) {
|
||||
List<Attribute> matches = new ArrayList<>();
|
||||
|
||||
// first take into account the qualified version
|
||||
|
@ -303,16 +302,15 @@ public class Analyzer extends RuleExecutor<LogicalPlan> {
|
|||
if (!a.expressionsResolved() && Resolvables.resolved(a.aggregates())) {
|
||||
List<Expression> groupings = a.groupings();
|
||||
List<Expression> newGroupings = new ArrayList<>();
|
||||
List<Attribute> resolved = Expressions.asAttributes(a.aggregates());
|
||||
AttributeMap<Expression> resolved = Expressions.asAttributeMap(a.aggregates());
|
||||
boolean changed = false;
|
||||
for (int i = 0; i < groupings.size(); i++) {
|
||||
Expression grouping = groupings.get(i);
|
||||
for (Expression grouping : groupings) {
|
||||
if (grouping instanceof UnresolvedAttribute) {
|
||||
Attribute maybeResolved = resolveAgainstList((UnresolvedAttribute) grouping, resolved);
|
||||
Attribute maybeResolved = resolveAgainstList((UnresolvedAttribute) grouping, resolved.keySet());
|
||||
if (maybeResolved != null) {
|
||||
changed = true;
|
||||
// use the matched expression (not its attribute)
|
||||
grouping = a.aggregates().get(i);
|
||||
grouping = resolved.get(maybeResolved);
|
||||
}
|
||||
}
|
||||
newGroupings.add(grouping);
|
||||
|
@ -883,10 +881,10 @@ public class Analyzer extends RuleExecutor<LogicalPlan> {
|
|||
};
|
||||
|
||||
//
|
||||
// Handle aggs in HAVING and ORDER BY clause. To help folding any aggs not found in Aggregation
|
||||
// Handle aggs in HAVING. To help folding any aggs not found in Aggregation
|
||||
// will be pushed down to the Aggregate and then projected. This also simplifies the Verifier's job.
|
||||
//
|
||||
private class ResolveAggsInHavingAndOrderBy extends AnalyzeRule<LogicalPlan> {
|
||||
private class ResolveAggsInHaving extends AnalyzeRule<LogicalPlan> {
|
||||
|
||||
@Override
|
||||
protected boolean skipResolved() {
|
||||
|
|
|
@ -353,7 +353,7 @@ abstract class Verifier {
|
|||
|
||||
return true;
|
||||
} else if (e instanceof Score) {
|
||||
// Score can't be an aggretate function
|
||||
// Score can't be an aggregate function
|
||||
missing.put(e, source);
|
||||
return true;
|
||||
}
|
||||
|
|
|
@ -7,6 +7,7 @@ package org.elasticsearch.xpack.sql.execution;
|
|||
|
||||
import org.elasticsearch.action.ActionListener;
|
||||
import org.elasticsearch.client.Client;
|
||||
import org.elasticsearch.common.io.stream.NamedWriteableRegistry;
|
||||
import org.elasticsearch.search.builder.SearchSourceBuilder;
|
||||
import org.elasticsearch.xpack.sql.analysis.analyzer.PreAnalyzer;
|
||||
import org.elasticsearch.xpack.sql.analysis.index.IndexResolver;
|
||||
|
@ -27,6 +28,7 @@ import java.util.List;
|
|||
|
||||
public class PlanExecutor {
|
||||
private final Client client;
|
||||
private final NamedWriteableRegistry writableRegistry;
|
||||
|
||||
private final FunctionRegistry functionRegistry;
|
||||
|
||||
|
@ -35,8 +37,10 @@ public class PlanExecutor {
|
|||
private final Optimizer optimizer;
|
||||
private final Planner planner;
|
||||
|
||||
public PlanExecutor(Client client, IndexResolver indexResolver) {
|
||||
public PlanExecutor(Client client, IndexResolver indexResolver, NamedWriteableRegistry writeableRegistry) {
|
||||
this.client = client;
|
||||
this.writableRegistry = writeableRegistry;
|
||||
|
||||
this.indexResolver = indexResolver;
|
||||
this.functionRegistry = new FunctionRegistry();
|
||||
|
||||
|
@ -45,6 +49,10 @@ public class PlanExecutor {
|
|||
this.planner = new Planner();
|
||||
}
|
||||
|
||||
public NamedWriteableRegistry writableRegistry() {
|
||||
return writableRegistry;
|
||||
}
|
||||
|
||||
private SqlSession newSession(Configuration cfg) {
|
||||
return new SqlSession(cfg, client, functionRegistry, indexResolver, preAnalyzer, optimizer, planner);
|
||||
}
|
||||
|
@ -65,7 +73,7 @@ public class PlanExecutor {
|
|||
}
|
||||
|
||||
public void nextPage(Configuration cfg, Cursor cursor, ActionListener<RowSet> listener) {
|
||||
cursor.nextPage(cfg, client, listener);
|
||||
cursor.nextPage(cfg, client, writableRegistry, listener);
|
||||
}
|
||||
|
||||
public void cleanCursor(Configuration cfg, Cursor cursor, ActionListener<Boolean> listener) {
|
||||
|
|
|
@ -0,0 +1,22 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License;
|
||||
* you may not use this file except in compliance with the Elastic License.
|
||||
*/
|
||||
package org.elasticsearch.xpack.sql.execution.search;
|
||||
|
||||
/**
|
||||
* Reference to a ES aggregation (which can be either a GROUP BY or Metric agg).
|
||||
*/
|
||||
public abstract class AggRef implements FieldExtraction {
|
||||
|
||||
@Override
|
||||
public void collectFields(SqlSourceBuilder sourceBuilder) {
|
||||
// Aggregations do not need any special fields
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean supportedByAggsOnlyQuery() {
|
||||
return true;
|
||||
}
|
||||
}
|
|
@ -1,133 +0,0 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License;
|
||||
* you may not use this file except in compliance with the Elastic License.
|
||||
*/
|
||||
package org.elasticsearch.xpack.sql.execution.search;
|
||||
|
||||
import java.util.Arrays;
|
||||
import java.util.List;
|
||||
|
||||
/**
|
||||
* Aggregations are returned by Elasticsearch in a tree structure where each
|
||||
* nested level can have a different size. For example a group by a, b, c
|
||||
* results in 3-level nested array where each level contains all the relevant
|
||||
* values for its parent entry.
|
||||
* <p>
|
||||
* Assuming there's a total of 2 A's, 3 B's and 5 C's, the values will be
|
||||
* <ul>
|
||||
* <li>A-agg level = { A1, A2 }
|
||||
* <li>B-agg level = { { A1B1, A1B2, A1B3 }, { A2B1, A2B2, A2B3 }
|
||||
* <li>C-agg level = { { { A1B1C1, A1B1C2 ..}, { A1B2C1, etc... } } } and so on
|
||||
* </ul>
|
||||
* <p>
|
||||
* Further more the columns are added in the order in which they are requested
|
||||
* (0, 1, 2) eliminating the need for keys as these are implicit (their position
|
||||
* in the list).
|
||||
* <p>
|
||||
* To help with the iteration, there are two dedicated counters:
|
||||
* one that carries (increments) the counter for each level (indicated by the
|
||||
* position inside the array) once the children reach their max a flat cursor
|
||||
* to indicate the row.
|
||||
*/
|
||||
class AggValues {
|
||||
private int row = 0;
|
||||
|
||||
private final List<Object[]> columns;
|
||||
private int[] indexPerLevel;
|
||||
private int size;
|
||||
|
||||
AggValues(List<Object[]> columns) {
|
||||
this.columns = columns;
|
||||
}
|
||||
|
||||
void init(int maxDepth, int limit) {
|
||||
int sz = computeSize(columns, maxDepth);
|
||||
size = limit > 0 ? Math.min(limit, sz) : sz;
|
||||
indexPerLevel = new int[maxDepth + 1];
|
||||
}
|
||||
|
||||
private static int computeSize(List<Object[]> columns, int maxDepth) {
|
||||
// look only at arrays with the right depth (the others might be
|
||||
// counters or other functions)
|
||||
// then return the parent array to compute the actual returned results
|
||||
Object[] leafArray = null;
|
||||
for (int i = 0; i < columns.size() && leafArray == null; i++) {
|
||||
Object[] col = columns.get(i);
|
||||
Object o = col;
|
||||
int level = 0;
|
||||
Object[] parent = null;
|
||||
// keep unwrapping until the desired level is reached
|
||||
while (o instanceof Object[]) {
|
||||
col = ((Object[]) o);
|
||||
if (col.length > 0) {
|
||||
if (level == maxDepth) {
|
||||
leafArray = parent;
|
||||
break;
|
||||
} else {
|
||||
parent = col;
|
||||
level++;
|
||||
o = col[0];
|
||||
}
|
||||
} else {
|
||||
o = null;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (leafArray == null) {
|
||||
return columns.get(0).length;
|
||||
}
|
||||
|
||||
int sz = 0;
|
||||
for (Object leaf : leafArray) {
|
||||
sz += ((Object[]) leaf).length;
|
||||
}
|
||||
return sz;
|
||||
}
|
||||
|
||||
Object column(int column) {
|
||||
Object o = columns.get(column);
|
||||
|
||||
for (int lvl = 0; o instanceof Object[]; lvl++) {
|
||||
Object[] arr = (Object[]) o;
|
||||
// the current branch is done
|
||||
if (indexPerLevel[lvl] == arr.length) {
|
||||
// reset the current branch
|
||||
indexPerLevel[lvl] = 0;
|
||||
// bump the parent - if it's too big it, the loop will restart
|
||||
// again from that position
|
||||
indexPerLevel[lvl - 1]++;
|
||||
// restart the loop
|
||||
lvl = -1;
|
||||
o = columns.get(column);
|
||||
} else {
|
||||
o = arr[indexPerLevel[lvl]];
|
||||
}
|
||||
}
|
||||
return o;
|
||||
}
|
||||
|
||||
int size() {
|
||||
return size;
|
||||
}
|
||||
|
||||
void reset() {
|
||||
row = 0;
|
||||
Arrays.fill(indexPerLevel, 0);
|
||||
}
|
||||
|
||||
boolean nextRow() {
|
||||
if (row < size - 1) {
|
||||
row++;
|
||||
// increment leaf counter - the size check is done lazily while retrieving the columns
|
||||
indexPerLevel[indexPerLevel.length - 1]++;
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
boolean hasCurrentRow() {
|
||||
return row < size;
|
||||
}
|
||||
}
|
|
@ -1,62 +0,0 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License;
|
||||
* you may not use this file except in compliance with the Elastic License.
|
||||
*/
|
||||
package org.elasticsearch.xpack.sql.execution.search;
|
||||
|
||||
import org.elasticsearch.xpack.sql.session.AbstractRowSet;
|
||||
import org.elasticsearch.xpack.sql.session.Cursor;
|
||||
import org.elasticsearch.xpack.sql.session.SchemaRowSet;
|
||||
import org.elasticsearch.xpack.sql.type.Schema;
|
||||
|
||||
import java.util.List;
|
||||
import java.util.function.Supplier;
|
||||
|
||||
class AggsRowSet extends AbstractRowSet implements SchemaRowSet {
|
||||
private final Schema schema;
|
||||
private final AggValues agg;
|
||||
private final List<Supplier<Object>> columns;
|
||||
private int row = 0;
|
||||
|
||||
AggsRowSet(Schema schema, AggValues agg, List<Supplier<Object>> columns) {
|
||||
this.schema = schema;
|
||||
this.agg = agg;
|
||||
this.columns = columns;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected Object getColumn(int column) {
|
||||
return columns.get(column).get();
|
||||
}
|
||||
|
||||
@Override
|
||||
protected boolean doHasCurrent() {
|
||||
return row < size();
|
||||
}
|
||||
|
||||
@Override
|
||||
protected boolean doNext() {
|
||||
return agg.nextRow();
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void doReset() {
|
||||
agg.reset();
|
||||
}
|
||||
|
||||
@Override
|
||||
public int size() {
|
||||
return agg.size();
|
||||
}
|
||||
|
||||
@Override
|
||||
public Cursor nextPageCursor() {
|
||||
return Cursor.EMPTY;
|
||||
}
|
||||
|
||||
@Override
|
||||
public Schema schema() {
|
||||
return schema;
|
||||
}
|
||||
}
|
|
@ -0,0 +1,208 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License;
|
||||
* you may not use this file except in compliance with the Elastic License.
|
||||
*/
|
||||
package org.elasticsearch.xpack.sql.execution.search;
|
||||
|
||||
import org.apache.logging.log4j.Logger;
|
||||
import org.elasticsearch.action.ActionListener;
|
||||
import org.elasticsearch.action.search.SearchRequest;
|
||||
import org.elasticsearch.action.search.SearchResponse;
|
||||
import org.elasticsearch.client.Client;
|
||||
import org.elasticsearch.common.bytes.BytesReference;
|
||||
import org.elasticsearch.common.io.stream.BytesStreamOutput;
|
||||
import org.elasticsearch.common.io.stream.NamedWriteableAwareStreamInput;
|
||||
import org.elasticsearch.common.io.stream.NamedWriteableRegistry;
|
||||
import org.elasticsearch.common.io.stream.StreamInput;
|
||||
import org.elasticsearch.common.io.stream.StreamOutput;
|
||||
import org.elasticsearch.common.logging.Loggers;
|
||||
import org.elasticsearch.search.aggregations.Aggregation;
|
||||
import org.elasticsearch.search.aggregations.AggregationBuilder;
|
||||
import org.elasticsearch.search.aggregations.bucket.composite.CompositeAggregation;
|
||||
import org.elasticsearch.search.aggregations.bucket.composite.CompositeAggregationBuilder;
|
||||
import org.elasticsearch.search.builder.SearchSourceBuilder;
|
||||
import org.elasticsearch.xpack.sql.SqlIllegalArgumentException;
|
||||
import org.elasticsearch.xpack.sql.execution.search.extractor.BucketExtractor;
|
||||
import org.elasticsearch.xpack.sql.querydsl.agg.Aggs;
|
||||
import org.elasticsearch.xpack.sql.session.Configuration;
|
||||
import org.elasticsearch.xpack.sql.session.Cursor;
|
||||
import org.elasticsearch.xpack.sql.session.RowSet;
|
||||
import org.elasticsearch.xpack.sql.util.StringUtils;
|
||||
|
||||
import java.io.IOException;
|
||||
import java.util.Arrays;
|
||||
import java.util.List;
|
||||
import java.util.Map;
|
||||
import java.util.Objects;
|
||||
|
||||
/**
|
||||
* Cursor for composite aggregation (GROUP BY).
|
||||
* Stores the query that gets updated/slides across requests.
|
||||
*/
|
||||
public class CompositeAggregationCursor implements Cursor {
|
||||
|
||||
private final Logger log = Loggers.getLogger(getClass());
|
||||
|
||||
public static final String NAME = "c";
|
||||
|
||||
private final String[] indices;
|
||||
private final byte[] nextQuery;
|
||||
private final List<BucketExtractor> extractors;
|
||||
private final int limit;
|
||||
|
||||
CompositeAggregationCursor(byte[] next, List<BucketExtractor> exts, int remainingLimit, String... indices) {
|
||||
this.indices = indices;
|
||||
this.nextQuery = next;
|
||||
this.extractors = exts;
|
||||
this.limit = remainingLimit;
|
||||
}
|
||||
|
||||
public CompositeAggregationCursor(StreamInput in) throws IOException {
|
||||
indices = in.readStringArray();
|
||||
nextQuery = in.readByteArray();
|
||||
limit = in.readVInt();
|
||||
|
||||
extractors = in.readNamedWriteableList(BucketExtractor.class);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void writeTo(StreamOutput out) throws IOException {
|
||||
out.writeStringArray(indices);
|
||||
out.writeByteArray(nextQuery);
|
||||
out.writeVInt(limit);
|
||||
|
||||
out.writeNamedWriteableList(extractors);
|
||||
}
|
||||
|
||||
@Override
|
||||
public String getWriteableName() {
|
||||
return NAME;
|
||||
}
|
||||
|
||||
String[] indices() {
|
||||
return indices;
|
||||
}
|
||||
|
||||
byte[] next() {
|
||||
return nextQuery;
|
||||
}
|
||||
|
||||
List<BucketExtractor> extractors() {
|
||||
return extractors;
|
||||
}
|
||||
|
||||
int limit() {
|
||||
return limit;
|
||||
}
|
||||
|
||||
@Override
|
||||
public void nextPage(Configuration cfg, Client client, NamedWriteableRegistry registry, ActionListener<RowSet> listener) {
|
||||
SearchSourceBuilder q;
|
||||
try {
|
||||
q = deserializeQuery(registry, nextQuery);
|
||||
} catch (Exception ex) {
|
||||
listener.onFailure(ex);
|
||||
return;
|
||||
}
|
||||
|
||||
SearchSourceBuilder query = q;
|
||||
if (log.isTraceEnabled()) {
|
||||
log.trace("About to execute composite query {} on {}", StringUtils.toString(query), indices);
|
||||
}
|
||||
|
||||
SearchRequest search = Querier.prepareRequest(client, query, cfg.pageTimeout(), indices);
|
||||
|
||||
client.search(search, ActionListener.wrap(r -> {
|
||||
updateCompositeAfterKey(r, query);
|
||||
CompositeAggsRowSet rowSet = new CompositeAggsRowSet(extractors, r, limit,
|
||||
serializeQuery(query), indices);
|
||||
listener.onResponse(rowSet);
|
||||
}, listener::onFailure));
|
||||
}
|
||||
|
||||
static CompositeAggregation getComposite(SearchResponse response) {
|
||||
Aggregation agg = response.getAggregations().get(Aggs.ROOT_GROUP_NAME);
|
||||
if (agg == null) {
|
||||
return null;
|
||||
}
|
||||
|
||||
if (agg instanceof CompositeAggregation) {
|
||||
return (CompositeAggregation) agg;
|
||||
}
|
||||
|
||||
throw new SqlIllegalArgumentException("Unrecognized root group found; {}", agg.getClass());
|
||||
}
|
||||
|
||||
static void updateCompositeAfterKey(SearchResponse r, SearchSourceBuilder next) {
|
||||
CompositeAggregation composite = getComposite(r);
|
||||
|
||||
if (composite == null) {
|
||||
throw new SqlIllegalArgumentException("Invalid server response; no group-by detected");
|
||||
}
|
||||
|
||||
Map<String, Object> afterKey = composite.afterKey();
|
||||
// a null after-key means done
|
||||
if (afterKey != null) {
|
||||
AggregationBuilder aggBuilder = next.aggregations().getAggregatorFactories().get(0);
|
||||
// update after-key with the new value
|
||||
if (aggBuilder instanceof CompositeAggregationBuilder) {
|
||||
CompositeAggregationBuilder comp = (CompositeAggregationBuilder) aggBuilder;
|
||||
comp.aggregateAfter(afterKey);
|
||||
} else {
|
||||
throw new SqlIllegalArgumentException("Invalid client request; expected a group-by but instead got {}", aggBuilder);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Deserializes the search source from a byte array.
|
||||
*/
|
||||
static SearchSourceBuilder deserializeQuery(NamedWriteableRegistry registry, byte[] source) throws IOException {
|
||||
try (NamedWriteableAwareStreamInput in = new NamedWriteableAwareStreamInput(StreamInput.wrap(source), registry)) {
|
||||
return new SearchSourceBuilder(in);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Serializes the search source to a byte array.
|
||||
*/
|
||||
static byte[] serializeQuery(SearchSourceBuilder source) throws IOException {
|
||||
if (source == null) {
|
||||
return new byte[0];
|
||||
}
|
||||
|
||||
try (BytesStreamOutput out = new BytesStreamOutput()) {
|
||||
source.writeTo(out);
|
||||
return BytesReference.toBytes(out.bytes());
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@Override
|
||||
public void clear(Configuration cfg, Client client, ActionListener<Boolean> listener) {
|
||||
listener.onResponse(true);
|
||||
}
|
||||
|
||||
@Override
|
||||
public int hashCode() {
|
||||
return Objects.hash(Arrays.hashCode(indices), Arrays.hashCode(nextQuery), extractors, limit);
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean equals(Object obj) {
|
||||
if (obj == null || obj.getClass() != getClass()) {
|
||||
return false;
|
||||
}
|
||||
CompositeAggregationCursor other = (CompositeAggregationCursor) obj;
|
||||
return Arrays.equals(indices, other.indices)
|
||||
&& Arrays.equals(nextQuery, other.nextQuery)
|
||||
&& Objects.equals(extractors, other.extractors)
|
||||
&& Objects.equals(limit, other.limit);
|
||||
}
|
||||
|
||||
@Override
|
||||
public String toString() {
|
||||
return "cursor for composite on index [" + Arrays.toString(indices) + "]";
|
||||
}
|
||||
}
|
|
@ -0,0 +1,100 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License;
|
||||
* you may not use this file except in compliance with the Elastic License.
|
||||
*/
|
||||
package org.elasticsearch.xpack.sql.execution.search;
|
||||
|
||||
import org.elasticsearch.action.search.SearchResponse;
|
||||
import org.elasticsearch.search.aggregations.bucket.composite.CompositeAggregation;
|
||||
import org.elasticsearch.xpack.sql.execution.search.extractor.BucketExtractor;
|
||||
import org.elasticsearch.xpack.sql.session.AbstractRowSet;
|
||||
import org.elasticsearch.xpack.sql.session.Cursor;
|
||||
import org.elasticsearch.xpack.sql.session.RowSet;
|
||||
|
||||
import java.util.List;
|
||||
|
||||
import static java.util.Collections.emptyList;
|
||||
|
||||
/**
|
||||
* {@link RowSet} specific to (GROUP BY) aggregation.
|
||||
*/
|
||||
class CompositeAggsRowSet extends AbstractRowSet {
|
||||
private final List<BucketExtractor> exts;
|
||||
|
||||
private final List<? extends CompositeAggregation.Bucket> buckets;
|
||||
|
||||
private final Cursor cursor;
|
||||
|
||||
private final int size;
|
||||
private int row = 0;
|
||||
|
||||
CompositeAggsRowSet(List<BucketExtractor> exts, SearchResponse response, int limit, byte[] next, String... indices) {
|
||||
this.exts = exts;
|
||||
|
||||
CompositeAggregation composite = CompositeAggregationCursor.getComposite(response);
|
||||
if (composite != null) {
|
||||
buckets = composite.getBuckets();
|
||||
} else {
|
||||
buckets = emptyList();
|
||||
}
|
||||
|
||||
// page size
|
||||
size = limit < 0 ? buckets.size() : Math.min(buckets.size(), limit);
|
||||
|
||||
if (next == null) {
|
||||
cursor = Cursor.EMPTY;
|
||||
} else {
|
||||
// compute remaining limit
|
||||
int remainingLimit = limit - size;
|
||||
// if the computed limit is zero, or the size is zero it means either there's nothing left or the limit has been reached
|
||||
// note that a composite agg might be valid but return zero groups (since these can be filtered with HAVING/bucket selector)
|
||||
// however the Querier takes care of that and keeps making requests until either the query is invalid or at least one response
|
||||
// is returned
|
||||
if (next == null || size == 0 || remainingLimit == 0) {
|
||||
cursor = Cursor.EMPTY;
|
||||
} else {
|
||||
cursor = new CompositeAggregationCursor(next, exts, remainingLimit, indices);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
protected Object getColumn(int column) {
|
||||
return exts.get(column).extract(buckets.get(row));
|
||||
}
|
||||
|
||||
@Override
|
||||
public int columnCount() {
|
||||
return exts.size();
|
||||
}
|
||||
|
||||
@Override
|
||||
protected boolean doHasCurrent() {
|
||||
return row < size;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected boolean doNext() {
|
||||
if (row < size - 1) {
|
||||
row++;
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void doReset() {
|
||||
row = 0;
|
||||
}
|
||||
|
||||
@Override
|
||||
public int size() {
|
||||
return size;
|
||||
}
|
||||
|
||||
@Override
|
||||
public Cursor nextPageCursor() {
|
||||
return cursor;
|
||||
}
|
||||
}
|
|
@ -24,13 +24,4 @@ public interface FieldExtraction {
|
|||
* ({@code true}) or should it force a scroll query ({@code false})?
|
||||
*/
|
||||
boolean supportedByAggsOnlyQuery();
|
||||
|
||||
/**
|
||||
* Indicates the depth of the result. Used for counting the actual size of a
|
||||
* result by knowing how many nested levels there are. Typically used by
|
||||
* aggregations.
|
||||
*
|
||||
* @return depth of the result
|
||||
*/
|
||||
int depth();
|
||||
}
|
||||
|
|
|
@ -0,0 +1,437 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License;
|
||||
* you may not use this file except in compliance with the Elastic License.
|
||||
*/
|
||||
package org.elasticsearch.xpack.sql.execution.search;
|
||||
|
||||
import org.apache.logging.log4j.Logger;
|
||||
import org.elasticsearch.action.ActionListener;
|
||||
import org.elasticsearch.action.search.SearchRequest;
|
||||
import org.elasticsearch.action.search.SearchResponse;
|
||||
import org.elasticsearch.action.search.ShardSearchFailure;
|
||||
import org.elasticsearch.client.Client;
|
||||
import org.elasticsearch.common.Nullable;
|
||||
import org.elasticsearch.common.logging.Loggers;
|
||||
import org.elasticsearch.common.unit.TimeValue;
|
||||
import org.elasticsearch.common.util.CollectionUtils;
|
||||
import org.elasticsearch.common.xcontent.XContentBuilder;
|
||||
import org.elasticsearch.index.query.QueryBuilder;
|
||||
import org.elasticsearch.search.SearchHit;
|
||||
import org.elasticsearch.search.aggregations.Aggregation;
|
||||
import org.elasticsearch.search.aggregations.Aggregations;
|
||||
import org.elasticsearch.search.aggregations.bucket.MultiBucketsAggregation.Bucket;
|
||||
import org.elasticsearch.search.aggregations.bucket.filter.Filters;
|
||||
import org.elasticsearch.search.builder.SearchSourceBuilder;
|
||||
import org.elasticsearch.xpack.sql.SqlIllegalArgumentException;
|
||||
import org.elasticsearch.xpack.sql.execution.search.extractor.BucketExtractor;
|
||||
import org.elasticsearch.xpack.sql.execution.search.extractor.CompositeKeyExtractor;
|
||||
import org.elasticsearch.xpack.sql.execution.search.extractor.ComputingExtractor;
|
||||
import org.elasticsearch.xpack.sql.execution.search.extractor.ConstantExtractor;
|
||||
import org.elasticsearch.xpack.sql.execution.search.extractor.FieldHitExtractor;
|
||||
import org.elasticsearch.xpack.sql.execution.search.extractor.HitExtractor;
|
||||
import org.elasticsearch.xpack.sql.execution.search.extractor.MetricAggExtractor;
|
||||
import org.elasticsearch.xpack.sql.expression.function.scalar.processor.definition.AggExtractorInput;
|
||||
import org.elasticsearch.xpack.sql.expression.function.scalar.processor.definition.AggPathInput;
|
||||
import org.elasticsearch.xpack.sql.expression.function.scalar.processor.definition.HitExtractorInput;
|
||||
import org.elasticsearch.xpack.sql.expression.function.scalar.processor.definition.ProcessorDefinition;
|
||||
import org.elasticsearch.xpack.sql.expression.function.scalar.processor.definition.ReferenceInput;
|
||||
import org.elasticsearch.xpack.sql.querydsl.agg.Aggs;
|
||||
import org.elasticsearch.xpack.sql.querydsl.container.ComputedRef;
|
||||
import org.elasticsearch.xpack.sql.querydsl.container.GlobalCountRef;
|
||||
import org.elasticsearch.xpack.sql.querydsl.container.GroupByRef;
|
||||
import org.elasticsearch.xpack.sql.querydsl.container.MetricAggRef;
|
||||
import org.elasticsearch.xpack.sql.querydsl.container.QueryContainer;
|
||||
import org.elasticsearch.xpack.sql.querydsl.container.ScriptFieldRef;
|
||||
import org.elasticsearch.xpack.sql.querydsl.container.SearchHitFieldRef;
|
||||
import org.elasticsearch.xpack.sql.session.Configuration;
|
||||
import org.elasticsearch.xpack.sql.session.Rows;
|
||||
import org.elasticsearch.xpack.sql.session.SchemaRowSet;
|
||||
import org.elasticsearch.xpack.sql.type.Schema;
|
||||
import org.elasticsearch.xpack.sql.util.StringUtils;
|
||||
|
||||
import java.io.IOException;
|
||||
import java.util.ArrayList;
|
||||
import java.util.LinkedHashSet;
|
||||
import java.util.List;
|
||||
import java.util.Set;
|
||||
|
||||
import static java.util.Collections.singletonList;
|
||||
// TODO: add retry/back-off
|
||||
public class Querier {
|
||||
|
||||
private final Logger log = Loggers.getLogger(getClass());
|
||||
|
||||
private final TimeValue keepAlive, timeout;
|
||||
private final int size;
|
||||
private final Client client;
|
||||
@Nullable
|
||||
private final QueryBuilder filter;
|
||||
|
||||
public Querier(Client client, Configuration cfg) {
|
||||
this(client, cfg.requestTimeout(), cfg.pageTimeout(), cfg.filter(), cfg.pageSize());
|
||||
}
|
||||
|
||||
public Querier(Client client, TimeValue keepAlive, TimeValue timeout, QueryBuilder filter, int size) {
|
||||
this.client = client;
|
||||
this.keepAlive = keepAlive;
|
||||
this.timeout = timeout;
|
||||
this.filter = filter;
|
||||
this.size = size;
|
||||
}
|
||||
|
||||
public void query(Schema schema, QueryContainer query, String index, ActionListener<SchemaRowSet> listener) {
|
||||
// prepare the request
|
||||
SearchSourceBuilder sourceBuilder = SourceGenerator.sourceBuilder(query, filter, size);
|
||||
// set query timeout
|
||||
if (timeout.getSeconds() > 0) {
|
||||
sourceBuilder.timeout(timeout);
|
||||
}
|
||||
|
||||
if (log.isTraceEnabled()) {
|
||||
log.trace("About to execute query {} on {}", StringUtils.toString(sourceBuilder), index);
|
||||
}
|
||||
|
||||
SearchRequest search = prepareRequest(client, sourceBuilder, timeout, index);
|
||||
|
||||
ActionListener<SearchResponse> l;
|
||||
if (query.isAggsOnly()) {
|
||||
if (query.aggs().useImplicitGroupBy()) {
|
||||
l = new ImplicitGroupActionListener(listener, client, timeout, schema, query, search);
|
||||
} else {
|
||||
l = new CompositeActionListener(listener, client, timeout, schema, query, search);
|
||||
}
|
||||
} else {
|
||||
search.scroll(keepAlive);
|
||||
l = new ScrollActionListener(listener, client, timeout, schema, query);
|
||||
}
|
||||
|
||||
client.search(search, l);
|
||||
}
|
||||
|
||||
public static SearchRequest prepareRequest(Client client, SearchSourceBuilder source, TimeValue timeout, String... indices) {
|
||||
SearchRequest search = client.prepareSearch(indices).setSource(source).setTimeout(timeout).request();
|
||||
search.allowPartialSearchResults(false);
|
||||
return search;
|
||||
}
|
||||
|
||||
/**
|
||||
* Dedicated listener for implicit/default group-by queries that return only _one_ result.
|
||||
*/
|
||||
static class ImplicitGroupActionListener extends BaseAggActionListener {
|
||||
|
||||
private static List<? extends Bucket> EMPTY_BUCKET = singletonList(new Bucket() {
|
||||
|
||||
@Override
|
||||
public XContentBuilder toXContent(XContentBuilder builder, Params params) throws IOException {
|
||||
throw new SqlIllegalArgumentException("No group-by/aggs defined");
|
||||
}
|
||||
|
||||
@Override
|
||||
public Object getKey() {
|
||||
throw new SqlIllegalArgumentException("No group-by/aggs defined");
|
||||
}
|
||||
|
||||
@Override
|
||||
public String getKeyAsString() {
|
||||
throw new SqlIllegalArgumentException("No group-by/aggs defined");
|
||||
}
|
||||
|
||||
@Override
|
||||
public long getDocCount() {
|
||||
throw new SqlIllegalArgumentException("No group-by/aggs defined");
|
||||
}
|
||||
|
||||
@Override
|
||||
public Aggregations getAggregations() {
|
||||
throw new SqlIllegalArgumentException("No group-by/aggs defined");
|
||||
}
|
||||
});
|
||||
|
||||
ImplicitGroupActionListener(ActionListener<SchemaRowSet> listener, Client client, TimeValue keepAlive, Schema schema,
|
||||
QueryContainer query, SearchRequest request) {
|
||||
super(listener, client, keepAlive, schema, query, request);
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void handleResponse(SearchResponse response, ActionListener<SchemaRowSet> listener) {
|
||||
Aggregations aggs = response.getAggregations();
|
||||
if (aggs != null) {
|
||||
Aggregation agg = aggs.get(Aggs.ROOT_GROUP_NAME);
|
||||
if (agg instanceof Filters) {
|
||||
handleBuckets(((Filters) agg).getBuckets(), response);
|
||||
} else {
|
||||
throw new SqlIllegalArgumentException("Unrecognized root group found; {}", agg.getClass());
|
||||
}
|
||||
}
|
||||
// can happen when only a count is requested which is derived from the response
|
||||
else {
|
||||
handleBuckets(EMPTY_BUCKET, response);
|
||||
}
|
||||
}
|
||||
|
||||
private void handleBuckets(List<? extends Bucket> buckets, SearchResponse response) {
|
||||
if (buckets.size() == 1) {
|
||||
Bucket implicitGroup = buckets.get(0);
|
||||
List<BucketExtractor> extractors = initBucketExtractors(response);
|
||||
Object[] values = new Object[extractors.size()];
|
||||
for (int i = 0; i < values.length; i++) {
|
||||
values[i] = extractors.get(i).extract(implicitGroup);
|
||||
}
|
||||
listener.onResponse(Rows.singleton(schema, values));
|
||||
|
||||
} else if (buckets.isEmpty()) {
|
||||
listener.onResponse(Rows.empty(schema));
|
||||
|
||||
} else {
|
||||
throw new SqlIllegalArgumentException("Too many groups returned by the implicit group; expected 1, received {}",
|
||||
buckets.size());
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Dedicated listener for composite aggs/group-by results.
|
||||
*/
|
||||
static class CompositeActionListener extends BaseAggActionListener {
|
||||
|
||||
CompositeActionListener(ActionListener<SchemaRowSet> listener, Client client, TimeValue keepAlive,
|
||||
Schema schema, QueryContainer query, SearchRequest request) {
|
||||
super(listener, client, keepAlive, schema, query, request);
|
||||
}
|
||||
|
||||
|
||||
@Override
|
||||
protected void handleResponse(SearchResponse response, ActionListener<SchemaRowSet> listener) {
|
||||
// there are some results
|
||||
if (response.getAggregations().asList().size() > 0) {
|
||||
CompositeAggregationCursor.updateCompositeAfterKey(response, request.source());
|
||||
|
||||
byte[] nextSearch = null;
|
||||
try {
|
||||
nextSearch = CompositeAggregationCursor.serializeQuery(request.source());
|
||||
} catch (Exception ex) {
|
||||
listener.onFailure(ex);
|
||||
return;
|
||||
}
|
||||
|
||||
listener.onResponse(
|
||||
new SchemaCompositeAggsRowSet(schema, initBucketExtractors(response), response, query.limit(),
|
||||
nextSearch,
|
||||
request.indices()));
|
||||
}
|
||||
// no results
|
||||
else {
|
||||
listener.onResponse(Rows.empty(schema));
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
abstract static class BaseAggActionListener extends BaseActionListener {
|
||||
final QueryContainer query;
|
||||
final SearchRequest request;
|
||||
|
||||
BaseAggActionListener(ActionListener<SchemaRowSet> listener, Client client, TimeValue keepAlive, Schema schema,
|
||||
QueryContainer query, SearchRequest request) {
|
||||
super(listener, client, keepAlive, schema);
|
||||
|
||||
this.query = query;
|
||||
this.request = request;
|
||||
}
|
||||
|
||||
protected List<BucketExtractor> initBucketExtractors(SearchResponse response) {
|
||||
// create response extractors for the first time
|
||||
List<FieldExtraction> refs = query.columns();
|
||||
|
||||
List<BucketExtractor> exts = new ArrayList<>(refs.size());
|
||||
for (FieldExtraction ref : refs) {
|
||||
exts.add(createExtractor(ref, new ConstantExtractor(response.getHits().getTotalHits())));
|
||||
}
|
||||
return exts;
|
||||
}
|
||||
|
||||
private BucketExtractor createExtractor(FieldExtraction ref, BucketExtractor totalCount) {
|
||||
if (ref instanceof GroupByRef) {
|
||||
GroupByRef r = (GroupByRef) ref;
|
||||
return new CompositeKeyExtractor(r.key(), r.property(), r.timeZone());
|
||||
}
|
||||
|
||||
if (ref instanceof MetricAggRef) {
|
||||
MetricAggRef r = (MetricAggRef) ref;
|
||||
return new MetricAggExtractor(r.name(), r.property(), r.innerKey());
|
||||
}
|
||||
|
||||
if (ref == GlobalCountRef.INSTANCE) {
|
||||
return totalCount;
|
||||
}
|
||||
|
||||
if (ref instanceof ComputedRef) {
|
||||
ProcessorDefinition proc = ((ComputedRef) ref).processor();
|
||||
|
||||
// wrap only agg inputs
|
||||
proc = proc.transformDown(l -> {
|
||||
BucketExtractor be = createExtractor(l.context(), totalCount);
|
||||
return new AggExtractorInput(l.location(), l.expression(), l.action(), be);
|
||||
}, AggPathInput.class);
|
||||
|
||||
return new ComputingExtractor(proc.asProcessor());
|
||||
}
|
||||
|
||||
throw new SqlIllegalArgumentException("Unexpected value reference {}", ref.getClass());
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Dedicated listener for column retrieval/non-grouped queries (scrolls).
|
||||
*/
|
||||
static class ScrollActionListener extends BaseActionListener {
|
||||
private final QueryContainer query;
|
||||
|
||||
ScrollActionListener(ActionListener<SchemaRowSet> listener, Client client, TimeValue keepAlive,
|
||||
Schema schema, QueryContainer query) {
|
||||
super(listener, client, keepAlive, schema);
|
||||
this.query = query;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void handleResponse(SearchResponse response, ActionListener<SchemaRowSet> listener) {
|
||||
SearchHit[] hits = response.getHits().getHits();
|
||||
|
||||
// create response extractors for the first time
|
||||
List<FieldExtraction> refs = query.columns();
|
||||
|
||||
List<HitExtractor> exts = new ArrayList<>(refs.size());
|
||||
for (FieldExtraction ref : refs) {
|
||||
exts.add(createExtractor(ref));
|
||||
}
|
||||
|
||||
// there are some results
|
||||
if (hits.length > 0) {
|
||||
String scrollId = response.getScrollId();
|
||||
|
||||
// if there's an id, try to setup next scroll
|
||||
if (scrollId != null &&
|
||||
// is all the content already retrieved?
|
||||
(Boolean.TRUE.equals(response.isTerminatedEarly()) || response.getHits().getTotalHits() == hits.length
|
||||
// or maybe the limit has been reached
|
||||
|| (hits.length >= query.limit() && query.limit() > -1))) {
|
||||
// if so, clear the scroll
|
||||
clear(response.getScrollId(), ActionListener.wrap(
|
||||
succeeded -> listener.onResponse(new SchemaSearchHitRowSet(schema, exts, hits, query.limit(), null)),
|
||||
listener::onFailure));
|
||||
} else {
|
||||
listener.onResponse(new SchemaSearchHitRowSet(schema, exts, hits, query.limit(), scrollId));
|
||||
}
|
||||
}
|
||||
// no hits
|
||||
else {
|
||||
clear(response.getScrollId(), ActionListener.wrap(succeeded -> listener.onResponse(Rows.empty(schema)),
|
||||
listener::onFailure));
|
||||
}
|
||||
}
|
||||
|
||||
private HitExtractor createExtractor(FieldExtraction ref) {
|
||||
if (ref instanceof SearchHitFieldRef) {
|
||||
SearchHitFieldRef f = (SearchHitFieldRef) ref;
|
||||
return new FieldHitExtractor(f.name(), f.useDocValue(), f.hitName());
|
||||
}
|
||||
|
||||
if (ref instanceof ScriptFieldRef) {
|
||||
ScriptFieldRef f = (ScriptFieldRef) ref;
|
||||
return new FieldHitExtractor(f.name(), true);
|
||||
}
|
||||
|
||||
if (ref instanceof ComputedRef) {
|
||||
ProcessorDefinition proc = ((ComputedRef) ref).processor();
|
||||
// collect hitNames
|
||||
Set<String> hitNames = new LinkedHashSet<>();
|
||||
proc = proc.transformDown(l -> {
|
||||
HitExtractor he = createExtractor(l.context());
|
||||
hitNames.add(he.hitName());
|
||||
|
||||
if (hitNames.size() > 1) {
|
||||
throw new SqlIllegalArgumentException("Multi-level nested fields [{}] not supported yet", hitNames);
|
||||
}
|
||||
|
||||
return new HitExtractorInput(l.location(), l.expression(), he);
|
||||
}, ReferenceInput.class);
|
||||
String hitName = null;
|
||||
if (hitNames.size() == 1) {
|
||||
hitName = hitNames.iterator().next();
|
||||
}
|
||||
return new ComputingExtractor(proc.asProcessor(), hitName);
|
||||
}
|
||||
|
||||
throw new SqlIllegalArgumentException("Unexpected value reference {}", ref.getClass());
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Base listener class providing clean-up and exception handling.
|
||||
* Handles both scroll queries (scan/scroll) and regular/composite-aggs queries.
|
||||
*/
|
||||
abstract static class BaseActionListener implements ActionListener<SearchResponse> {
|
||||
|
||||
final ActionListener<SchemaRowSet> listener;
|
||||
|
||||
final Client client;
|
||||
final TimeValue keepAlive;
|
||||
final Schema schema;
|
||||
|
||||
BaseActionListener(ActionListener<SchemaRowSet> listener, Client client, TimeValue keepAlive, Schema schema) {
|
||||
this.listener = listener;
|
||||
|
||||
this.client = client;
|
||||
this.keepAlive = keepAlive;
|
||||
this.schema = schema;
|
||||
}
|
||||
|
||||
// TODO: need to handle rejections plus check failures (shard size, etc...)
|
||||
@Override
|
||||
public void onResponse(final SearchResponse response) {
|
||||
try {
|
||||
ShardSearchFailure[] failure = response.getShardFailures();
|
||||
if (!CollectionUtils.isEmpty(failure)) {
|
||||
cleanup(response, new SqlIllegalArgumentException(failure[0].reason(), failure[0].getCause()));
|
||||
} else {
|
||||
handleResponse(response, ActionListener.wrap(listener::onResponse, e -> cleanup(response, e)));
|
||||
}
|
||||
} catch (Exception ex) {
|
||||
cleanup(response, ex);
|
||||
}
|
||||
}
|
||||
|
||||
protected abstract void handleResponse(SearchResponse response, ActionListener<SchemaRowSet> listener);
|
||||
|
||||
// clean-up the scroll in case of exception
|
||||
protected final void cleanup(SearchResponse response, Exception ex) {
|
||||
if (response != null && response.getScrollId() != null) {
|
||||
client.prepareClearScroll().addScrollId(response.getScrollId())
|
||||
// in case of failure, report the initial exception instead of the one resulting from cleaning the scroll
|
||||
.execute(ActionListener.wrap(r -> listener.onFailure(ex), e -> {
|
||||
ex.addSuppressed(e);
|
||||
listener.onFailure(ex);
|
||||
}));
|
||||
} else {
|
||||
listener.onFailure(ex);
|
||||
}
|
||||
}
|
||||
|
||||
protected final void clear(String scrollId, ActionListener<Boolean> listener) {
|
||||
if (scrollId != null) {
|
||||
client.prepareClearScroll().addScrollId(scrollId).execute(
|
||||
ActionListener.wrap(
|
||||
clearScrollResponse -> listener.onResponse(clearScrollResponse.isSucceeded()),
|
||||
listener::onFailure));
|
||||
} else {
|
||||
listener.onResponse(false);
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public final void onFailure(Exception ex) {
|
||||
listener.onFailure(ex);
|
||||
}
|
||||
}
|
||||
}
|
|
@ -0,0 +1,34 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License;
|
||||
* you may not use this file except in compliance with the Elastic License.
|
||||
*/
|
||||
package org.elasticsearch.xpack.sql.execution.search;
|
||||
|
||||
import org.elasticsearch.action.search.SearchResponse;
|
||||
import org.elasticsearch.xpack.sql.execution.search.extractor.BucketExtractor;
|
||||
import org.elasticsearch.xpack.sql.session.RowSet;
|
||||
import org.elasticsearch.xpack.sql.session.SchemaRowSet;
|
||||
import org.elasticsearch.xpack.sql.type.Schema;
|
||||
|
||||
import java.util.List;
|
||||
|
||||
/**
|
||||
* Extension of the {@link RowSet} over a composite agg, extending it to provide its schema.
|
||||
* Used for the initial response.
|
||||
*/
|
||||
class SchemaCompositeAggsRowSet extends CompositeAggsRowSet implements SchemaRowSet {
|
||||
|
||||
private final Schema schema;
|
||||
|
||||
SchemaCompositeAggsRowSet(Schema schema, List<BucketExtractor> exts, SearchResponse response, int limitAggs, byte[] next,
|
||||
String... indices) {
|
||||
super(exts, response, limitAggs, next, indices);
|
||||
this.schema = schema;
|
||||
}
|
||||
|
||||
@Override
|
||||
public Schema schema() {
|
||||
return schema;
|
||||
}
|
||||
}
|
|
@ -14,13 +14,13 @@ import java.util.List;
|
|||
|
||||
/**
|
||||
* Initial results from a scroll search. Distinct from the following pages
|
||||
* because it has a {@link Schema} available. See {@link ScrolledSearchHitRowSet}
|
||||
* because it has a {@link Schema} available. See {@link SearchHitRowSet}
|
||||
* for the next pages.
|
||||
*/
|
||||
public class InitialSearchHitRowSet extends AbstractSearchHitRowSet implements SchemaRowSet {
|
||||
class SchemaSearchHitRowSet extends SearchHitRowSet implements SchemaRowSet {
|
||||
private final Schema schema;
|
||||
|
||||
public InitialSearchHitRowSet(Schema schema, List<HitExtractor> exts, SearchHit[] hits, int limitHits, String scrollId) {
|
||||
SchemaSearchHitRowSet(Schema schema, List<HitExtractor> exts, SearchHit[] hits, int limitHits, String scrollId) {
|
||||
super(exts, hits, limitHits, scrollId);
|
||||
this.schema = schema;
|
||||
}
|
|
@ -6,14 +6,17 @@
|
|||
package org.elasticsearch.xpack.sql.execution.search;
|
||||
|
||||
|
||||
import org.apache.logging.log4j.Logger;
|
||||
import org.elasticsearch.action.ActionListener;
|
||||
import org.elasticsearch.action.search.ClearScrollRequest;
|
||||
import org.elasticsearch.action.search.ClearScrollResponse;
|
||||
import org.elasticsearch.action.search.SearchResponse;
|
||||
import org.elasticsearch.action.search.SearchScrollRequest;
|
||||
import org.elasticsearch.client.Client;
|
||||
import org.elasticsearch.common.io.stream.NamedWriteableRegistry;
|
||||
import org.elasticsearch.common.io.stream.StreamInput;
|
||||
import org.elasticsearch.common.io.stream.StreamOutput;
|
||||
import org.elasticsearch.common.logging.Loggers;
|
||||
import org.elasticsearch.xpack.sql.execution.search.extractor.HitExtractor;
|
||||
import org.elasticsearch.xpack.sql.session.Configuration;
|
||||
import org.elasticsearch.xpack.sql.session.Cursor;
|
||||
|
@ -24,6 +27,9 @@ import java.util.List;
|
|||
import java.util.Objects;
|
||||
|
||||
public class ScrollCursor implements Cursor {
|
||||
|
||||
private final Logger log = Loggers.getLogger(getClass());
|
||||
|
||||
public static final String NAME = "s";
|
||||
|
||||
private final String scrollId;
|
||||
|
@ -38,28 +44,42 @@ public class ScrollCursor implements Cursor {
|
|||
|
||||
public ScrollCursor(StreamInput in) throws IOException {
|
||||
scrollId = in.readString();
|
||||
extractors = in.readNamedWriteableList(HitExtractor.class);
|
||||
limit = in.readVInt();
|
||||
|
||||
extractors = in.readNamedWriteableList(HitExtractor.class);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void writeTo(StreamOutput out) throws IOException {
|
||||
out.writeString(scrollId);
|
||||
out.writeNamedWriteableList(extractors);
|
||||
out.writeVInt(limit);
|
||||
}
|
||||
|
||||
out.writeNamedWriteableList(extractors);
|
||||
}
|
||||
|
||||
@Override
|
||||
public String getWriteableName() {
|
||||
return NAME;
|
||||
}
|
||||
|
||||
String scrollId() {
|
||||
return scrollId;
|
||||
}
|
||||
|
||||
List<HitExtractor> extractors() {
|
||||
return extractors;
|
||||
}
|
||||
|
||||
int limit() {
|
||||
return limit;
|
||||
}
|
||||
@Override
|
||||
public void nextPage(Configuration cfg, Client client, ActionListener<RowSet> listener) {
|
||||
public void nextPage(Configuration cfg, Client client, NamedWriteableRegistry registry, ActionListener<RowSet> listener) {
|
||||
log.trace("About to execute scroll query {}", scrollId);
|
||||
|
||||
SearchScrollRequest request = new SearchScrollRequest(scrollId).scroll(cfg.pageTimeout());
|
||||
client.searchScroll(request, ActionListener.wrap((SearchResponse response) -> {
|
||||
ScrolledSearchHitRowSet rowSet = new ScrolledSearchHitRowSet(extractors, response.getHits().getHits(),
|
||||
SearchHitRowSet rowSet = new SearchHitRowSet(extractors, response.getHits().getHits(),
|
||||
limit, response.getScrollId());
|
||||
if (rowSet.nextPageCursor() == Cursor.EMPTY ) {
|
||||
// we are finished with this cursor, let's clean it before continuing
|
||||
|
@ -104,5 +124,4 @@ public class ScrollCursor implements Cursor {
|
|||
clearScrollRequest.addScrollId(scrollId);
|
||||
client.clearScroll(clearScrollRequest, listener);
|
||||
}
|
||||
|
||||
}
|
|
@ -1,31 +0,0 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License;
|
||||
* you may not use this file except in compliance with the Elastic License.
|
||||
*/
|
||||
package org.elasticsearch.xpack.sql.execution.search;
|
||||
|
||||
import org.elasticsearch.search.SearchHit;
|
||||
import org.elasticsearch.xpack.sql.execution.search.extractor.HitExtractor;
|
||||
import org.elasticsearch.xpack.sql.type.Schema;
|
||||
|
||||
import java.util.List;
|
||||
|
||||
/**
|
||||
* "Next" page of results from a scroll search. Distinct from the first page
|
||||
* because it no longer has the {@link Schema}. See {@link InitialSearchHitRowSet}
|
||||
* for the initial results.
|
||||
*/
|
||||
public class ScrolledSearchHitRowSet extends AbstractSearchHitRowSet {
|
||||
private final int columnCount;
|
||||
|
||||
public ScrolledSearchHitRowSet(List<HitExtractor> exts, SearchHit[] hits, int limitHits, String scrollId) {
|
||||
super(exts, hits, limitHits, scrollId);
|
||||
this.columnCount = exts.size();
|
||||
}
|
||||
|
||||
@Override
|
||||
public int columnCount() {
|
||||
return columnCount;
|
||||
}
|
||||
}
|
|
@ -1,379 +0,0 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License;
|
||||
* you may not use this file except in compliance with the Elastic License.
|
||||
*/
|
||||
package org.elasticsearch.xpack.sql.execution.search;
|
||||
|
||||
import org.apache.logging.log4j.Logger;
|
||||
import org.elasticsearch.action.ActionListener;
|
||||
import org.elasticsearch.action.search.SearchRequest;
|
||||
import org.elasticsearch.action.search.SearchResponse;
|
||||
import org.elasticsearch.action.search.ShardSearchFailure;
|
||||
import org.elasticsearch.client.Client;
|
||||
import org.elasticsearch.common.Nullable;
|
||||
import org.elasticsearch.common.logging.Loggers;
|
||||
import org.elasticsearch.common.unit.TimeValue;
|
||||
import org.elasticsearch.common.util.CollectionUtils;
|
||||
import org.elasticsearch.index.query.QueryBuilder;
|
||||
import org.elasticsearch.search.SearchHit;
|
||||
import org.elasticsearch.search.aggregations.Aggregations;
|
||||
import org.elasticsearch.search.aggregations.InternalAggregation;
|
||||
import org.elasticsearch.search.aggregations.bucket.MultiBucketsAggregation;
|
||||
import org.elasticsearch.search.aggregations.bucket.MultiBucketsAggregation.Bucket;
|
||||
import org.elasticsearch.search.aggregations.support.AggregationPath;
|
||||
import org.elasticsearch.search.builder.SearchSourceBuilder;
|
||||
import org.elasticsearch.xpack.sql.SqlException;
|
||||
import org.elasticsearch.xpack.sql.SqlIllegalArgumentException;
|
||||
import org.elasticsearch.xpack.sql.execution.search.extractor.ComputingHitExtractor;
|
||||
import org.elasticsearch.xpack.sql.execution.search.extractor.FieldHitExtractor;
|
||||
import org.elasticsearch.xpack.sql.execution.search.extractor.HitExtractor;
|
||||
import org.elasticsearch.xpack.sql.expression.function.scalar.processor.definition.AggPathInput;
|
||||
import org.elasticsearch.xpack.sql.expression.function.scalar.processor.definition.AggValueInput;
|
||||
import org.elasticsearch.xpack.sql.expression.function.scalar.processor.definition.HitExtractorInput;
|
||||
import org.elasticsearch.xpack.sql.expression.function.scalar.processor.definition.ProcessorDefinition;
|
||||
import org.elasticsearch.xpack.sql.expression.function.scalar.processor.definition.ReferenceInput;
|
||||
import org.elasticsearch.xpack.sql.expression.function.scalar.processor.runtime.Processor;
|
||||
import org.elasticsearch.xpack.sql.querydsl.agg.AggPath;
|
||||
import org.elasticsearch.xpack.sql.querydsl.container.AggRef;
|
||||
import org.elasticsearch.xpack.sql.querydsl.container.ComputedRef;
|
||||
import org.elasticsearch.xpack.sql.querydsl.container.QueryContainer;
|
||||
import org.elasticsearch.xpack.sql.querydsl.container.ScriptFieldRef;
|
||||
import org.elasticsearch.xpack.sql.querydsl.container.SearchHitFieldRef;
|
||||
import org.elasticsearch.xpack.sql.querydsl.container.TotalCountRef;
|
||||
import org.elasticsearch.xpack.sql.session.Configuration;
|
||||
import org.elasticsearch.xpack.sql.session.Rows;
|
||||
import org.elasticsearch.xpack.sql.session.SchemaRowSet;
|
||||
import org.elasticsearch.xpack.sql.type.Schema;
|
||||
import org.elasticsearch.xpack.sql.util.StringUtils;
|
||||
|
||||
import java.util.ArrayList;
|
||||
import java.util.LinkedHashSet;
|
||||
import java.util.List;
|
||||
import java.util.Map;
|
||||
import java.util.Set;
|
||||
import java.util.function.Supplier;
|
||||
// TODO: add retry/back-off
|
||||
public class Scroller {
|
||||
|
||||
private final Logger log = Loggers.getLogger(getClass());
|
||||
|
||||
private final TimeValue keepAlive, timeout;
|
||||
private final int size;
|
||||
private final Client client;
|
||||
@Nullable
|
||||
private final QueryBuilder filter;
|
||||
|
||||
public Scroller(Client client, Configuration cfg) {
|
||||
this(client, cfg.requestTimeout(), cfg.pageTimeout(), cfg.filter(), cfg.pageSize());
|
||||
}
|
||||
|
||||
public Scroller(Client client, TimeValue keepAlive, TimeValue timeout, QueryBuilder filter, int size) {
|
||||
this.client = client;
|
||||
this.keepAlive = keepAlive;
|
||||
this.timeout = timeout;
|
||||
this.filter = filter;
|
||||
this.size = size;
|
||||
}
|
||||
|
||||
public void scroll(Schema schema, QueryContainer query, String index, ActionListener<SchemaRowSet> listener) {
|
||||
// prepare the request
|
||||
SearchSourceBuilder sourceBuilder = SourceGenerator.sourceBuilder(query, filter, size);
|
||||
|
||||
if (log.isTraceEnabled()) {
|
||||
log.trace("About to execute query {} on {}", StringUtils.toString(sourceBuilder), index);
|
||||
}
|
||||
|
||||
SearchRequest search = client.prepareSearch(index).setSource(sourceBuilder).request();
|
||||
|
||||
ScrollerActionListener l;
|
||||
if (query.isAggsOnly()) {
|
||||
l = new AggsScrollActionListener(listener, client, timeout, schema, query);
|
||||
} else {
|
||||
search.scroll(keepAlive).source().timeout(timeout);
|
||||
l = new HandshakeScrollActionListener(listener, client, timeout, schema, query);
|
||||
}
|
||||
client.search(search, l);
|
||||
}
|
||||
|
||||
// dedicated scroll used for aggs-only/group-by results
|
||||
static class AggsScrollActionListener extends ScrollerActionListener {
|
||||
|
||||
private final QueryContainer query;
|
||||
|
||||
AggsScrollActionListener(ActionListener<SchemaRowSet> listener, Client client, TimeValue keepAlive,
|
||||
Schema schema, QueryContainer query) {
|
||||
super(listener, client, keepAlive, schema);
|
||||
this.query = query;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void handleResponse(SearchResponse response, ActionListener<SchemaRowSet> listener) {
|
||||
|
||||
final List<Object[]> extractedAggs = new ArrayList<>();
|
||||
AggValues aggValues = new AggValues(extractedAggs);
|
||||
List<Supplier<Object>> aggColumns = new ArrayList<>(query.columns().size());
|
||||
|
||||
// this method assumes the nested aggregation are all part of the same tree (the SQL group-by)
|
||||
int maxDepth = -1;
|
||||
|
||||
List<FieldExtraction> cols = query.columns();
|
||||
for (int index = 0; index < cols.size(); index++) {
|
||||
FieldExtraction col = cols.get(index);
|
||||
Supplier<Object> supplier = null;
|
||||
|
||||
if (col instanceof ComputedRef) {
|
||||
ComputedRef pRef = (ComputedRef) col;
|
||||
|
||||
Processor processor = pRef.processor().transformUp(a -> {
|
||||
Object[] value = extractAggValue(new AggRef(a.context()), response);
|
||||
extractedAggs.add(value);
|
||||
final int aggPosition = extractedAggs.size() - 1;
|
||||
Supplier<Object> action = null;
|
||||
if (a.action() != null) {
|
||||
action = () -> a.action().process(aggValues.column(aggPosition));
|
||||
}
|
||||
else {
|
||||
action = () -> aggValues.column(aggPosition);
|
||||
}
|
||||
return new AggValueInput(a.location(), a.expression(), action, a.innerKey());
|
||||
}, AggPathInput.class).asProcessor();
|
||||
// the input is provided through the value input above
|
||||
supplier = () -> processor.process(null);
|
||||
}
|
||||
else {
|
||||
extractedAggs.add(extractAggValue(col, response));
|
||||
final int aggPosition = extractedAggs.size() - 1;
|
||||
supplier = () -> aggValues.column(aggPosition);
|
||||
}
|
||||
|
||||
aggColumns.add(supplier);
|
||||
if (col.depth() > maxDepth) {
|
||||
maxDepth = col.depth();
|
||||
}
|
||||
}
|
||||
|
||||
aggValues.init(maxDepth, query.limit());
|
||||
clearScroll(response.getScrollId(), ActionListener.wrap(
|
||||
succeeded -> listener.onResponse(new AggsRowSet(schema, aggValues, aggColumns)),
|
||||
listener::onFailure));
|
||||
}
|
||||
|
||||
private Object[] extractAggValue(FieldExtraction col, SearchResponse response) {
|
||||
if (col == TotalCountRef.INSTANCE) {
|
||||
return new Object[] { Long.valueOf(response.getHits().getTotalHits()) };
|
||||
}
|
||||
else if (col instanceof AggRef) {
|
||||
Object[] arr;
|
||||
AggRef ref = (AggRef) col;
|
||||
|
||||
String path = ref.path();
|
||||
// yup, this is instance equality to make sure we only check the path used by the code
|
||||
if (path == TotalCountRef.PATH) {
|
||||
arr = new Object[] { Long.valueOf(response.getHits().getTotalHits()) };
|
||||
}
|
||||
else {
|
||||
// workaround for elastic/elasticsearch/issues/23056
|
||||
boolean formattedKey = AggPath.isBucketValueFormatted(path);
|
||||
if (formattedKey) {
|
||||
path = AggPath.bucketValueWithoutFormat(path);
|
||||
}
|
||||
Object value = getAggProperty(response.getAggregations(), path);
|
||||
|
||||
// unwrap nested map
|
||||
if (ref.innerKey() != null) {
|
||||
// needs changing when moving to Composite
|
||||
if (value instanceof Object[]) {
|
||||
Object[] val = (Object[]) value;
|
||||
arr = new Object[val.length];
|
||||
for (int i = 0; i < arr.length; i++) {
|
||||
if (val[i] instanceof Map) {
|
||||
arr[i] = ((Map<?, ?>) val[i]).get(ref.innerKey());
|
||||
}
|
||||
}
|
||||
value = arr;
|
||||
} else {
|
||||
if (value instanceof Map) {
|
||||
value = new Object[] { ((Map<?, ?>) value).get(ref.innerKey()) };
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (formattedKey) {
|
||||
List<? extends Bucket> buckets = ((MultiBucketsAggregation) value).getBuckets();
|
||||
arr = new Object[buckets.size()];
|
||||
for (int i = 0; i < buckets.size(); i++) {
|
||||
arr[i] = buckets.get(i).getKeyAsString();
|
||||
}
|
||||
} else {
|
||||
arr = value instanceof Object[] ? (Object[]) value : new Object[] { value };
|
||||
}
|
||||
}
|
||||
|
||||
return arr;
|
||||
}
|
||||
throw new SqlIllegalArgumentException("Unexpected non-agg/grouped column specified; {}", col.getClass());
|
||||
}
|
||||
|
||||
private static Object getAggProperty(Aggregations aggs, String path) {
|
||||
List<String> list = AggregationPath.parse(path).getPathElementsAsStringList();
|
||||
String aggName = list.get(0);
|
||||
InternalAggregation agg = aggs.get(aggName);
|
||||
if (agg == null) {
|
||||
throw new SqlException("Cannot find an aggregation named {}", aggName);
|
||||
}
|
||||
return agg.getProperty(list.subList(1, list.size()));
|
||||
}
|
||||
}
|
||||
|
||||
// initial scroll used for parsing search hits (handles possible aggs)
|
||||
static class HandshakeScrollActionListener extends ScrollerActionListener {
|
||||
private final QueryContainer query;
|
||||
|
||||
HandshakeScrollActionListener(ActionListener<SchemaRowSet> listener, Client client, TimeValue keepAlive,
|
||||
Schema schema, QueryContainer query) {
|
||||
super(listener, client, keepAlive, schema);
|
||||
this.query = query;
|
||||
}
|
||||
|
||||
|
||||
@Override
|
||||
protected void handleResponse(SearchResponse response, ActionListener<SchemaRowSet> listener) {
|
||||
SearchHit[] hits = response.getHits().getHits();
|
||||
List<HitExtractor> exts = getExtractors();
|
||||
|
||||
// there are some results
|
||||
if (hits.length > 0) {
|
||||
String scrollId = response.getScrollId();
|
||||
|
||||
// if there's an id, try to setup next scroll
|
||||
if (scrollId != null &&
|
||||
// is all the content already retrieved?
|
||||
(Boolean.TRUE.equals(response.isTerminatedEarly()) || response.getHits().getTotalHits() == hits.length
|
||||
// or maybe the limit has been reached
|
||||
|| (hits.length >= query.limit() && query.limit() > -1))) {
|
||||
// if so, clear the scroll
|
||||
clearScroll(response.getScrollId(), ActionListener.wrap(
|
||||
succeeded -> listener.onResponse(new InitialSearchHitRowSet(schema, exts, hits, query.limit(), null)),
|
||||
listener::onFailure));
|
||||
} else {
|
||||
listener.onResponse(new InitialSearchHitRowSet(schema, exts, hits, query.limit(), scrollId));
|
||||
}
|
||||
}
|
||||
// no hits
|
||||
else {
|
||||
clearScroll(response.getScrollId(), ActionListener.wrap(succeeded -> listener.onResponse(Rows.empty(schema)),
|
||||
listener::onFailure));
|
||||
}
|
||||
}
|
||||
|
||||
private List<HitExtractor> getExtractors() {
|
||||
// create response extractors for the first time
|
||||
List<FieldExtraction> refs = query.columns();
|
||||
|
||||
List<HitExtractor> exts = new ArrayList<>(refs.size());
|
||||
|
||||
for (FieldExtraction ref : refs) {
|
||||
exts.add(createExtractor(ref));
|
||||
}
|
||||
return exts;
|
||||
}
|
||||
|
||||
private HitExtractor createExtractor(FieldExtraction ref) {
|
||||
if (ref instanceof SearchHitFieldRef) {
|
||||
SearchHitFieldRef f = (SearchHitFieldRef) ref;
|
||||
return new FieldHitExtractor(f.name(), f.useDocValue(), f.hitName());
|
||||
}
|
||||
|
||||
if (ref instanceof ScriptFieldRef) {
|
||||
ScriptFieldRef f = (ScriptFieldRef) ref;
|
||||
return new FieldHitExtractor(f.name(), true);
|
||||
}
|
||||
|
||||
if (ref instanceof ComputedRef) {
|
||||
ProcessorDefinition proc = ((ComputedRef) ref).processor();
|
||||
// collect hitNames
|
||||
Set<String> hitNames = new LinkedHashSet<>();
|
||||
proc = proc.transformDown(l -> {
|
||||
HitExtractor he = createExtractor(l.context());
|
||||
hitNames.add(he.hitName());
|
||||
|
||||
if (hitNames.size() > 1) {
|
||||
throw new SqlIllegalArgumentException("Multi-level nested fields [{}] not supported yet", hitNames);
|
||||
}
|
||||
|
||||
return new HitExtractorInput(l.location(), l.expression(), he);
|
||||
}, ReferenceInput.class);
|
||||
String hitName = null;
|
||||
if (hitNames.size() == 1) {
|
||||
hitName = hitNames.iterator().next();
|
||||
}
|
||||
return new ComputingHitExtractor(proc.asProcessor(), hitName);
|
||||
}
|
||||
|
||||
throw new SqlIllegalArgumentException("Unexpected ValueReference {}", ref.getClass());
|
||||
}
|
||||
}
|
||||
|
||||
abstract static class ScrollerActionListener implements ActionListener<SearchResponse> {
|
||||
|
||||
final ActionListener<SchemaRowSet> listener;
|
||||
|
||||
final Client client;
|
||||
final TimeValue keepAlive;
|
||||
final Schema schema;
|
||||
|
||||
ScrollerActionListener(ActionListener<SchemaRowSet> listener, Client client, TimeValue keepAlive, Schema schema) {
|
||||
this.listener = listener;
|
||||
|
||||
this.client = client;
|
||||
this.keepAlive = keepAlive;
|
||||
this.schema = schema;
|
||||
}
|
||||
|
||||
// TODO: need to handle rejections plus check failures (shard size, etc...)
|
||||
@Override
|
||||
public void onResponse(final SearchResponse response) {
|
||||
try {
|
||||
ShardSearchFailure[] failure = response.getShardFailures();
|
||||
if (!CollectionUtils.isEmpty(failure)) {
|
||||
cleanupScroll(response, new SqlException(failure[0].reason(), failure[0].getCause()));
|
||||
} else {
|
||||
handleResponse(response, ActionListener.wrap(listener::onResponse, e -> cleanupScroll(response, e)));
|
||||
}
|
||||
} catch (Exception ex) {
|
||||
cleanupScroll(response, ex);
|
||||
}
|
||||
}
|
||||
|
||||
protected abstract void handleResponse(SearchResponse response, ActionListener<SchemaRowSet> listener);
|
||||
|
||||
// clean-up the scroll in case of exception
|
||||
protected final void cleanupScroll(SearchResponse response, Exception ex) {
|
||||
if (response != null && response.getScrollId() != null) {
|
||||
client.prepareClearScroll().addScrollId(response.getScrollId())
|
||||
// in case of failure, report the initial exception instead of the one resulting from cleaning the scroll
|
||||
.execute(ActionListener.wrap(r -> listener.onFailure(ex), e -> {
|
||||
ex.addSuppressed(e);
|
||||
listener.onFailure(ex);
|
||||
}));
|
||||
}
|
||||
}
|
||||
|
||||
protected final void clearScroll(String scrollId, ActionListener<Boolean> listener) {
|
||||
if (scrollId != null) {
|
||||
client.prepareClearScroll().addScrollId(scrollId).execute(
|
||||
ActionListener.wrap(
|
||||
clearScrollResponse -> listener.onResponse(clearScrollResponse.isSucceeded()),
|
||||
listener::onFailure));
|
||||
} else {
|
||||
listener.onResponse(false);
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public final void onFailure(Exception ex) {
|
||||
listener.onFailure(ex);
|
||||
}
|
||||
}
|
||||
}
|
|
@ -20,20 +20,20 @@ import java.util.Set;
|
|||
/**
|
||||
* Extracts rows from an array of {@link SearchHit}.
|
||||
*/
|
||||
abstract class AbstractSearchHitRowSet extends AbstractRowSet {
|
||||
class SearchHitRowSet extends AbstractRowSet {
|
||||
private final SearchHit[] hits;
|
||||
private final Cursor cursor;
|
||||
private final String scrollId;
|
||||
private final List<HitExtractor> extractors;
|
||||
private final Set<String> innerHits = new LinkedHashSet<>();
|
||||
private final String innerHit;
|
||||
private final int limit;
|
||||
|
||||
private final int size;
|
||||
private final int[] indexPerLevel;
|
||||
private int row = 0;
|
||||
|
||||
AbstractSearchHitRowSet(List<HitExtractor> exts, SearchHit[] hits, int limitHits, String scrollId) {
|
||||
SearchHitRowSet(List<HitExtractor> exts, SearchHit[] hits, int limit, String scrollId) {
|
||||
|
||||
this.hits = hits;
|
||||
this.scrollId = scrollId;
|
||||
this.extractors = exts;
|
||||
|
@ -60,8 +60,7 @@ abstract class AbstractSearchHitRowSet extends AbstractRowSet {
|
|||
maxDepth = 1;
|
||||
|
||||
sz = 0;
|
||||
for (int i = 0; i < hits.length; i++) {
|
||||
SearchHit hit = hits[i];
|
||||
for (SearchHit hit : hits) {
|
||||
for (String ih : innerHits) {
|
||||
SearchHits sh = hit.getInnerHits().get(ih);
|
||||
if (sh != null) {
|
||||
|
@ -70,10 +69,8 @@ abstract class AbstractSearchHitRowSet extends AbstractRowSet {
|
|||
}
|
||||
}
|
||||
}
|
||||
// overall limit
|
||||
limit = limitHits;
|
||||
// page size
|
||||
size = limitHits < 0 ? sz : Math.min(sz, limitHits);
|
||||
size = limit < 0 ? sz : Math.min(sz, limit);
|
||||
indexPerLevel = new int[maxDepth + 1];
|
||||
this.innerHit = innerHit;
|
||||
|
||||
|
@ -82,8 +79,8 @@ abstract class AbstractSearchHitRowSet extends AbstractRowSet {
|
|||
* scroll but all results fit in the first page. */
|
||||
cursor = Cursor.EMPTY;
|
||||
} else {
|
||||
// compute remaining limit
|
||||
int remainingLimit = limit - size;
|
||||
// compute remaining limit (only if the limit is specified - that is, positive).
|
||||
int remainingLimit = limit < 0 ? limit : limit - size;
|
||||
// if the computed limit is zero, or the size is zero it means either there's nothing left or the limit has been reached
|
||||
if (size == 0 || remainingLimit == 0) {
|
||||
cursor = Cursor.EMPTY;
|
||||
|
@ -93,6 +90,11 @@ abstract class AbstractSearchHitRowSet extends AbstractRowSet {
|
|||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public int columnCount() {
|
||||
return extractors.size();
|
||||
}
|
||||
|
||||
@Override
|
||||
protected Object getColumn(int column) {
|
||||
HitExtractor e = extractors.get(column);
|
||||
|
@ -109,7 +111,7 @@ abstract class AbstractSearchHitRowSet extends AbstractRowSet {
|
|||
hit = sh[indexPerLevel[lvl]];
|
||||
}
|
||||
|
||||
return e.get(hit);
|
||||
return e.extract(hit);
|
||||
}
|
||||
|
||||
@Override
|
|
@ -9,7 +9,7 @@ import org.elasticsearch.index.query.BoolQueryBuilder;
|
|||
import org.elasticsearch.index.query.ConstantScoreQueryBuilder;
|
||||
import org.elasticsearch.index.query.QueryBuilder;
|
||||
import org.elasticsearch.search.aggregations.AggregationBuilder;
|
||||
import org.elasticsearch.search.aggregations.PipelineAggregationBuilder;
|
||||
import org.elasticsearch.search.aggregations.bucket.composite.CompositeAggregationBuilder;
|
||||
import org.elasticsearch.search.builder.SearchSourceBuilder;
|
||||
import org.elasticsearch.search.fetch.StoredFieldsContext;
|
||||
import org.elasticsearch.search.fetch.subphase.FetchSourceContext;
|
||||
|
@ -17,20 +17,14 @@ import org.elasticsearch.search.sort.FieldSortBuilder;
|
|||
import org.elasticsearch.search.sort.NestedSortBuilder;
|
||||
import org.elasticsearch.search.sort.ScriptSortBuilder.ScriptSortType;
|
||||
import org.elasticsearch.search.sort.SortBuilder;
|
||||
import org.elasticsearch.search.sort.SortOrder;
|
||||
import org.elasticsearch.xpack.sql.expression.Attribute;
|
||||
import org.elasticsearch.xpack.sql.expression.FieldAttribute;
|
||||
import org.elasticsearch.xpack.sql.querydsl.agg.Aggs;
|
||||
import org.elasticsearch.xpack.sql.querydsl.agg.GroupByColumnAgg;
|
||||
import org.elasticsearch.xpack.sql.querydsl.agg.GroupingAgg;
|
||||
import org.elasticsearch.xpack.sql.querydsl.container.AttributeSort;
|
||||
import org.elasticsearch.xpack.sql.querydsl.container.QueryContainer;
|
||||
import org.elasticsearch.xpack.sql.querydsl.container.ScoreSort;
|
||||
import org.elasticsearch.xpack.sql.querydsl.container.ScriptSort;
|
||||
import org.elasticsearch.xpack.sql.querydsl.container.Sort;
|
||||
import org.elasticsearch.xpack.sql.querydsl.container.Sort.Direction;
|
||||
|
||||
import java.util.ArrayList;
|
||||
import java.util.List;
|
||||
|
||||
import static java.util.Collections.singletonList;
|
||||
|
@ -50,58 +44,45 @@ public abstract class SourceGenerator {
|
|||
source.query(new ConstantScoreQueryBuilder(filter));
|
||||
}
|
||||
} else {
|
||||
if (filter == null) {
|
||||
source.query(container.query().asBuilder());
|
||||
} else {
|
||||
if (filter != null) {
|
||||
source.query(new BoolQueryBuilder().must(container.query().asBuilder()).filter(filter));
|
||||
} else {
|
||||
source.query(container.query().asBuilder());
|
||||
}
|
||||
}
|
||||
|
||||
SqlSourceBuilder sortBuilder = new SqlSourceBuilder();
|
||||
// Iterate through all the columns requested, collecting the fields that
|
||||
// need to be retrieved from the result documents
|
||||
|
||||
// NB: the sortBuilder takes care of eliminating duplicates
|
||||
container.columns().forEach(cr -> cr.collectFields(sortBuilder));
|
||||
sortBuilder.build(source);
|
||||
optimize(sortBuilder, source);
|
||||
|
||||
// add the aggs
|
||||
Aggs aggs = container.aggs();
|
||||
// add the aggs (if present)
|
||||
AggregationBuilder aggBuilder = container.aggs().asAggBuilder();
|
||||
|
||||
// push limit onto group aggs
|
||||
if (container.limit() > 0) {
|
||||
List<GroupingAgg> groups = new ArrayList<>(aggs.groups());
|
||||
if (groups.size() > 0) {
|
||||
// get just the root agg
|
||||
GroupingAgg mainAgg = groups.get(0);
|
||||
if (mainAgg instanceof GroupByColumnAgg) {
|
||||
groups.set(0, ((GroupByColumnAgg) mainAgg).withLimit(container.limit()));
|
||||
aggs = aggs.with(groups);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
for (AggregationBuilder builder : aggs.asAggBuilders()) {
|
||||
source.aggregation(builder);
|
||||
if (aggBuilder != null) {
|
||||
source.aggregation(aggBuilder);
|
||||
}
|
||||
|
||||
sorting(container, source);
|
||||
|
||||
// add the pipeline aggs
|
||||
for (PipelineAggregationBuilder builder : aggs.asPipelineBuilders()) {
|
||||
source.aggregation(builder);
|
||||
// set page size
|
||||
if (size != null) {
|
||||
int sz = container.limit() > 0 ? Math.min(container.limit(), size) : size;
|
||||
|
||||
if (source.size() == -1) {
|
||||
source.size(sz);
|
||||
}
|
||||
if (aggBuilder instanceof CompositeAggregationBuilder) {
|
||||
((CompositeAggregationBuilder) aggBuilder).size(sz);
|
||||
}
|
||||
}
|
||||
|
||||
optimize(container, source);
|
||||
|
||||
// set size
|
||||
if (size != null) {
|
||||
if (source.size() == -1) {
|
||||
int sz = container.limit() > 0 ? Math.min(container.limit(), size) : size;
|
||||
source.size(sz);
|
||||
}
|
||||
}
|
||||
|
||||
return source;
|
||||
}
|
||||
|
||||
|
@ -158,7 +139,7 @@ public abstract class SourceGenerator {
|
|||
}
|
||||
|
||||
if (sortBuilder != null) {
|
||||
sortBuilder.order(sortable.direction() == Direction.ASC ? SortOrder.ASC : SortOrder.DESC);
|
||||
sortBuilder.order(sortable.direction().asOrder());
|
||||
source.sort(sortBuilder);
|
||||
}
|
||||
}
|
||||
|
@ -171,9 +152,10 @@ public abstract class SourceGenerator {
|
|||
}
|
||||
|
||||
private static void optimize(QueryContainer query, SearchSourceBuilder builder) {
|
||||
// if only aggs are needed, don't retrieve any docs
|
||||
// if only aggs are needed, don't retrieve any docs and remove scoring
|
||||
if (query.isAggsOnly()) {
|
||||
builder.size(0);
|
||||
builder.trackScores(false);
|
||||
// disable source fetching (only doc values are used)
|
||||
disableSource(builder);
|
||||
}
|
||||
|
|
|
@ -0,0 +1,17 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License;
|
||||
* you may not use this file except in compliance with the Elastic License.
|
||||
*/
|
||||
package org.elasticsearch.xpack.sql.execution.search.extractor;
|
||||
|
||||
import org.elasticsearch.common.io.stream.NamedWriteable;
|
||||
import org.elasticsearch.search.aggregations.bucket.MultiBucketsAggregation.Bucket;
|
||||
|
||||
/**
|
||||
* Extracts an aggregation value from a {@link Bucket}.
|
||||
*/
|
||||
public interface BucketExtractor extends NamedWriteable {
|
||||
|
||||
Object extract(Bucket bucket);
|
||||
}
|
|
@ -0,0 +1,30 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License;
|
||||
* you may not use this file except in compliance with the Elastic License.
|
||||
*/
|
||||
package org.elasticsearch.xpack.sql.execution.search.extractor;
|
||||
|
||||
import org.elasticsearch.common.io.stream.NamedWriteableRegistry;
|
||||
import org.elasticsearch.common.io.stream.NamedWriteableRegistry.Entry;
|
||||
|
||||
import java.util.ArrayList;
|
||||
import java.util.List;
|
||||
|
||||
public final class BucketExtractors {
|
||||
|
||||
private BucketExtractors() {}
|
||||
|
||||
/**
|
||||
* All of the named writeables needed to deserialize the instances of
|
||||
* {@linkplain BucketExtractor}s.
|
||||
*/
|
||||
public static List<NamedWriteableRegistry.Entry> getNamedWriteables() {
|
||||
List<NamedWriteableRegistry.Entry> entries = new ArrayList<>();
|
||||
entries.add(new Entry(BucketExtractor.class, CompositeKeyExtractor.NAME, CompositeKeyExtractor::new));
|
||||
entries.add(new Entry(BucketExtractor.class, ComputingExtractor.NAME, ComputingExtractor::new));
|
||||
entries.add(new Entry(BucketExtractor.class, MetricAggExtractor.NAME, MetricAggExtractor::new));
|
||||
entries.add(new Entry(BucketExtractor.class, ConstantExtractor.NAME, ConstantExtractor::new));
|
||||
return entries;
|
||||
}
|
||||
}
|
|
@ -0,0 +1,121 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License;
|
||||
* you may not use this file except in compliance with the Elastic License.
|
||||
*/
|
||||
package org.elasticsearch.xpack.sql.execution.search.extractor;
|
||||
|
||||
import org.elasticsearch.common.io.stream.StreamInput;
|
||||
import org.elasticsearch.common.io.stream.StreamOutput;
|
||||
import org.elasticsearch.search.aggregations.bucket.MultiBucketsAggregation.Bucket;
|
||||
import org.elasticsearch.xpack.sql.SqlIllegalArgumentException;
|
||||
import org.elasticsearch.xpack.sql.querydsl.container.GroupByRef.Property;
|
||||
import org.joda.time.DateTime;
|
||||
import org.joda.time.DateTimeZone;
|
||||
|
||||
import java.io.IOException;
|
||||
import java.util.Map;
|
||||
import java.util.Objects;
|
||||
|
||||
public class CompositeKeyExtractor implements BucketExtractor {
|
||||
|
||||
/**
|
||||
* Key or Komposite extractor.
|
||||
*/
|
||||
static final String NAME = "k";
|
||||
|
||||
private final String key;
|
||||
private final Property property;
|
||||
private final DateTimeZone timeZone;
|
||||
|
||||
/**
|
||||
* Constructs a new <code>CompositeKeyExtractor</code> instance.
|
||||
* The time-zone parameter is used to indicate a date key.
|
||||
*/
|
||||
public CompositeKeyExtractor(String key, Property property, DateTimeZone timeZone) {
|
||||
this.key = key;
|
||||
this.property = property;
|
||||
this.timeZone = timeZone;
|
||||
}
|
||||
|
||||
CompositeKeyExtractor(StreamInput in) throws IOException {
|
||||
key = in.readString();
|
||||
property = in.readEnum(Property.class);
|
||||
timeZone = in.readOptionalTimeZone();
|
||||
}
|
||||
|
||||
@Override
|
||||
public void writeTo(StreamOutput out) throws IOException {
|
||||
out.writeString(key);
|
||||
out.writeEnum(property);
|
||||
out.writeOptionalTimeZone(timeZone);
|
||||
}
|
||||
|
||||
String key() {
|
||||
return key;
|
||||
}
|
||||
|
||||
Property property() {
|
||||
return property;
|
||||
}
|
||||
|
||||
DateTimeZone timeZone() {
|
||||
return timeZone;
|
||||
}
|
||||
|
||||
@Override
|
||||
public String getWriteableName() {
|
||||
return NAME;
|
||||
}
|
||||
|
||||
@Override
|
||||
public Object extract(Bucket bucket) {
|
||||
if (property == Property.COUNT) {
|
||||
return bucket.getDocCount();
|
||||
}
|
||||
// get the composite value
|
||||
Object m = bucket.getKey();
|
||||
|
||||
if (!(m instanceof Map)) {
|
||||
throw new SqlIllegalArgumentException("Unexpected bucket returned: {}", m);
|
||||
}
|
||||
|
||||
Object object = ((Map<?, ?>) m).get(key);
|
||||
|
||||
if (timeZone != null) {
|
||||
if (object instanceof Long) {
|
||||
object = new DateTime(((Long) object).longValue(), timeZone);
|
||||
} else {
|
||||
throw new SqlIllegalArgumentException("Invalid date key returned: {}", object);
|
||||
}
|
||||
}
|
||||
|
||||
return object;
|
||||
}
|
||||
|
||||
@Override
|
||||
public int hashCode() {
|
||||
return Objects.hash(key, property, timeZone);
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean equals(Object obj) {
|
||||
if (this == obj) {
|
||||
return true;
|
||||
}
|
||||
|
||||
if (obj == null || getClass() != obj.getClass()) {
|
||||
return false;
|
||||
}
|
||||
|
||||
CompositeKeyExtractor other = (CompositeKeyExtractor) obj;
|
||||
return Objects.equals(key, other.key)
|
||||
&& Objects.equals(property, other.property)
|
||||
&& Objects.equals(timeZone, other.timeZone);
|
||||
}
|
||||
|
||||
@Override
|
||||
public String toString() {
|
||||
return "|" + key + "|";
|
||||
}
|
||||
}
|
|
@ -8,6 +8,7 @@ package org.elasticsearch.xpack.sql.execution.search.extractor;
|
|||
import org.elasticsearch.common.io.stream.StreamInput;
|
||||
import org.elasticsearch.common.io.stream.StreamOutput;
|
||||
import org.elasticsearch.search.SearchHit;
|
||||
import org.elasticsearch.search.aggregations.bucket.MultiBucketsAggregation.Bucket;
|
||||
import org.elasticsearch.xpack.sql.expression.function.scalar.processor.runtime.HitExtractorProcessor;
|
||||
import org.elasticsearch.xpack.sql.expression.function.scalar.processor.runtime.Processor;
|
||||
|
||||
|
@ -15,7 +16,7 @@ import java.io.IOException;
|
|||
import java.util.Objects;
|
||||
|
||||
/**
|
||||
* HitExtractor that delegates to a processor. The difference between this class
|
||||
* Hit/BucketExtractor that delegates to a processor. The difference between this class
|
||||
* and {@link HitExtractorProcessor} is that the latter is used inside a
|
||||
* {@link Processor} tree as a leaf (and thus can effectively parse the
|
||||
* {@link SearchHit} while this class is used when scrolling and passing down
|
||||
|
@ -25,7 +26,7 @@ import java.util.Objects;
|
|||
* to reduce API complexity (and keep the {@link HitExtractor} only as an
|
||||
* internal implementation detail).
|
||||
*/
|
||||
public class ComputingHitExtractor implements HitExtractor {
|
||||
public class ComputingExtractor implements HitExtractor, BucketExtractor {
|
||||
/**
|
||||
* Stands for {@code comPuting}. We try to use short names for {@link HitExtractor}s
|
||||
* to save a few bytes when when we send them back to the user.
|
||||
|
@ -34,12 +35,16 @@ public class ComputingHitExtractor implements HitExtractor {
|
|||
private final Processor processor;
|
||||
private final String hitName;
|
||||
|
||||
public ComputingHitExtractor(Processor processor, String hitName) {
|
||||
public ComputingExtractor(Processor processor) {
|
||||
this(processor, null);
|
||||
}
|
||||
|
||||
public ComputingExtractor(Processor processor, String hitName) {
|
||||
this.processor = processor;
|
||||
this.hitName = hitName;
|
||||
}
|
||||
|
||||
ComputingHitExtractor(StreamInput in) throws IOException {
|
||||
ComputingExtractor(StreamInput in) throws IOException {
|
||||
processor = in.readNamedWriteable(Processor.class);
|
||||
hitName = in.readOptionalString();
|
||||
}
|
||||
|
@ -59,8 +64,17 @@ public class ComputingHitExtractor implements HitExtractor {
|
|||
return processor;
|
||||
}
|
||||
|
||||
public Object extract(Object input) {
|
||||
return processor.process(input);
|
||||
}
|
||||
|
||||
@Override
|
||||
public Object get(SearchHit hit) {
|
||||
public Object extract(Bucket bucket) {
|
||||
return processor.process(bucket);
|
||||
}
|
||||
|
||||
@Override
|
||||
public Object extract(SearchHit hit) {
|
||||
return processor.process(hit);
|
||||
}
|
||||
|
||||
|
@ -74,7 +88,7 @@ public class ComputingHitExtractor implements HitExtractor {
|
|||
if (obj == null || obj.getClass() != getClass()) {
|
||||
return false;
|
||||
}
|
||||
ComputingHitExtractor other = (ComputingHitExtractor) obj;
|
||||
ComputingExtractor other = (ComputingExtractor) obj;
|
||||
return Objects.equals(processor, other.processor)
|
||||
&& Objects.equals(hitName, other.hitName);
|
||||
}
|
|
@ -8,6 +8,7 @@ package org.elasticsearch.xpack.sql.execution.search.extractor;
|
|||
import org.elasticsearch.common.io.stream.StreamInput;
|
||||
import org.elasticsearch.common.io.stream.StreamOutput;
|
||||
import org.elasticsearch.search.SearchHit;
|
||||
import org.elasticsearch.search.aggregations.bucket.MultiBucketsAggregation.Bucket;
|
||||
|
||||
import java.io.IOException;
|
||||
import java.util.Objects;
|
||||
|
@ -15,7 +16,7 @@ import java.util.Objects;
|
|||
/**
|
||||
* Returns the a constant for every search hit against which it is run.
|
||||
*/
|
||||
public class ConstantExtractor implements HitExtractor {
|
||||
public class ConstantExtractor implements HitExtractor, BucketExtractor {
|
||||
/**
|
||||
* Stands for {@code constant}. We try to use short names for {@link HitExtractor}s
|
||||
* to save a few bytes when when we send them back to the user.
|
||||
|
@ -42,7 +43,12 @@ public class ConstantExtractor implements HitExtractor {
|
|||
}
|
||||
|
||||
@Override
|
||||
public Object get(SearchHit hit) {
|
||||
public Object extract(SearchHit hit) {
|
||||
return constant;
|
||||
}
|
||||
|
||||
@Override
|
||||
public Object extract(Bucket bucket) {
|
||||
return constant;
|
||||
}
|
||||
|
||||
|
|
|
@ -10,7 +10,6 @@ import org.elasticsearch.common.document.DocumentField;
|
|||
import org.elasticsearch.common.io.stream.StreamInput;
|
||||
import org.elasticsearch.common.io.stream.StreamOutput;
|
||||
import org.elasticsearch.search.SearchHit;
|
||||
import org.elasticsearch.xpack.sql.SqlException;
|
||||
import org.elasticsearch.xpack.sql.SqlIllegalArgumentException;
|
||||
import org.joda.time.ReadableDateTime;
|
||||
|
||||
|
@ -83,7 +82,7 @@ public class FieldHitExtractor implements HitExtractor {
|
|||
}
|
||||
|
||||
@Override
|
||||
public Object get(SearchHit hit) {
|
||||
public Object extract(SearchHit hit) {
|
||||
Object value = null;
|
||||
if (useDocValue) {
|
||||
DocumentField field = hit.field(fieldName);
|
||||
|
@ -111,18 +110,18 @@ public class FieldHitExtractor implements HitExtractor {
|
|||
if (ARRAYS_LENIENCY || list.size() == 1) {
|
||||
return unwrapMultiValue(list.get(0));
|
||||
} else {
|
||||
throw new SqlException("Arrays (returned by [{}]) are not supported", fieldName);
|
||||
throw new SqlIllegalArgumentException("Arrays (returned by [{}]) are not supported", fieldName);
|
||||
}
|
||||
}
|
||||
}
|
||||
if (values instanceof Map) {
|
||||
throw new SqlException("Objects (returned by [{}]) are not supported", fieldName);
|
||||
throw new SqlIllegalArgumentException("Objects (returned by [{}]) are not supported", fieldName);
|
||||
}
|
||||
if (values instanceof Long || values instanceof Double || values instanceof String || values instanceof Boolean
|
||||
|| values instanceof ReadableDateTime) {
|
||||
return values;
|
||||
}
|
||||
throw new SqlException("Type {} (returned by [{}]) is not supported", values.getClass().getSimpleName(), fieldName);
|
||||
throw new SqlIllegalArgumentException("Type {} (returned by [{}]) is not supported", values.getClass().getSimpleName(), fieldName);
|
||||
}
|
||||
|
||||
@SuppressWarnings("unchecked")
|
||||
|
@ -137,7 +136,7 @@ public class FieldHitExtractor implements HitExtractor {
|
|||
first = false;
|
||||
value = ((Map<String, Object>) value).get(node);
|
||||
} else {
|
||||
throw new SqlException("Cannot extract value [{}] from source", fieldName);
|
||||
throw new SqlIllegalArgumentException("Cannot extract value [{}] from source", fieldName);
|
||||
}
|
||||
}
|
||||
return unwrapMultiValue(value);
|
||||
|
|
|
@ -16,7 +16,7 @@ public interface HitExtractor extends NamedWriteable {
|
|||
/**
|
||||
* Extract the value from a hit.
|
||||
*/
|
||||
Object get(SearchHit hit);
|
||||
Object extract(SearchHit hit);
|
||||
|
||||
/**
|
||||
* Name of the inner hit needed by this extractor if it needs one, {@code null} otherwise.
|
||||
|
|
|
@ -7,12 +7,14 @@ package org.elasticsearch.xpack.sql.execution.search.extractor;
|
|||
|
||||
import org.elasticsearch.common.io.stream.NamedWriteableRegistry;
|
||||
import org.elasticsearch.common.io.stream.NamedWriteableRegistry.Entry;
|
||||
import org.elasticsearch.xpack.sql.expression.function.scalar.Processors;
|
||||
|
||||
import java.util.ArrayList;
|
||||
import java.util.List;
|
||||
|
||||
public abstract class HitExtractors {
|
||||
public final class HitExtractors {
|
||||
|
||||
private HitExtractors() {}
|
||||
|
||||
/**
|
||||
* All of the named writeables needed to deserialize the instances of
|
||||
* {@linkplain HitExtractor}.
|
||||
|
@ -21,9 +23,8 @@ public abstract class HitExtractors {
|
|||
List<NamedWriteableRegistry.Entry> entries = new ArrayList<>();
|
||||
entries.add(new Entry(HitExtractor.class, ConstantExtractor.NAME, ConstantExtractor::new));
|
||||
entries.add(new Entry(HitExtractor.class, FieldHitExtractor.NAME, FieldHitExtractor::new));
|
||||
entries.add(new Entry(HitExtractor.class, ComputingHitExtractor.NAME, ComputingHitExtractor::new));
|
||||
entries.add(new Entry(HitExtractor.class, ComputingExtractor.NAME, ComputingExtractor::new));
|
||||
entries.add(new Entry(HitExtractor.class, ScoreExtractor.NAME, in -> ScoreExtractor.INSTANCE));
|
||||
entries.addAll(Processors.getNamedWriteables());
|
||||
return entries;
|
||||
}
|
||||
}
|
||||
|
|
|
@ -0,0 +1,108 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License;
|
||||
* you may not use this file except in compliance with the Elastic License.
|
||||
*/
|
||||
package org.elasticsearch.xpack.sql.execution.search.extractor;
|
||||
|
||||
import org.elasticsearch.common.io.stream.StreamInput;
|
||||
import org.elasticsearch.common.io.stream.StreamOutput;
|
||||
import org.elasticsearch.search.aggregations.InternalAggregation;
|
||||
import org.elasticsearch.search.aggregations.bucket.MultiBucketsAggregation.Bucket;
|
||||
import org.elasticsearch.search.aggregations.metrics.InternalNumericMetricsAggregation;
|
||||
import org.elasticsearch.xpack.sql.SqlIllegalArgumentException;
|
||||
import org.elasticsearch.xpack.sql.querydsl.agg.Aggs;
|
||||
|
||||
import java.io.IOException;
|
||||
import java.util.Map;
|
||||
import java.util.Objects;
|
||||
|
||||
public class MetricAggExtractor implements BucketExtractor {
|
||||
|
||||
static final String NAME = "m";
|
||||
|
||||
private final String name;
|
||||
private final String property;
|
||||
private final String innerKey;
|
||||
|
||||
public MetricAggExtractor(String name, String property, String innerKey) {
|
||||
this.name = name;
|
||||
this.property = property;
|
||||
this.innerKey = innerKey;
|
||||
}
|
||||
|
||||
MetricAggExtractor(StreamInput in) throws IOException {
|
||||
name = in.readString();
|
||||
property = in.readString();
|
||||
innerKey = in.readOptionalString();
|
||||
}
|
||||
|
||||
@Override
|
||||
public void writeTo(StreamOutput out) throws IOException {
|
||||
out.writeString(name);
|
||||
out.writeString(property);
|
||||
out.writeOptionalString(innerKey);
|
||||
}
|
||||
|
||||
String name() {
|
||||
return name;
|
||||
}
|
||||
|
||||
String property() {
|
||||
return property;
|
||||
}
|
||||
|
||||
String innerKey() {
|
||||
return innerKey;
|
||||
}
|
||||
|
||||
@Override
|
||||
public String getWriteableName() {
|
||||
return NAME;
|
||||
}
|
||||
|
||||
@Override
|
||||
public Object extract(Bucket bucket) {
|
||||
InternalAggregation agg = bucket.getAggregations().get(name);
|
||||
if (agg == null) {
|
||||
throw new SqlIllegalArgumentException("Cannot find an aggregation named {}", name);
|
||||
}
|
||||
if (agg instanceof InternalNumericMetricsAggregation.MultiValue) {
|
||||
//TODO: need to investigate when this can be not-null
|
||||
//if (innerKey == null) {
|
||||
// throw new SqlIllegalArgumentException("Invalid innerKey {} specified for aggregation {}", innerKey, name);
|
||||
//}
|
||||
return ((InternalNumericMetricsAggregation.MultiValue) agg).value(property);
|
||||
}
|
||||
|
||||
Object v = agg.getProperty(property);
|
||||
return innerKey != null && v instanceof Map ? ((Map<?, ?>) v).get(innerKey) : v;
|
||||
}
|
||||
|
||||
@Override
|
||||
public int hashCode() {
|
||||
return Objects.hash(name, property, innerKey);
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean equals(Object obj) {
|
||||
if (this == obj) {
|
||||
return true;
|
||||
}
|
||||
|
||||
if (obj == null || getClass() != obj.getClass()) {
|
||||
return false;
|
||||
}
|
||||
|
||||
MetricAggExtractor other = (MetricAggExtractor) obj;
|
||||
return Objects.equals(name, other.name)
|
||||
&& Objects.equals(property, other.property)
|
||||
&& Objects.equals(innerKey, other.innerKey);
|
||||
}
|
||||
|
||||
@Override
|
||||
public String toString() {
|
||||
String i = innerKey != null ? "[" + innerKey + "]" : "";
|
||||
return Aggs.ROOT_GROUP_NAME + ">" + name + "." + property + i;
|
||||
}
|
||||
}
|
|
@ -34,7 +34,7 @@ public class ScoreExtractor implements HitExtractor {
|
|||
}
|
||||
|
||||
@Override
|
||||
public Object get(SearchHit hit) {
|
||||
public Object extract(SearchHit hit) {
|
||||
return hit.getScore();
|
||||
}
|
||||
|
||||
|
|
|
@ -13,6 +13,7 @@ import java.util.List;
|
|||
import java.util.function.Predicate;
|
||||
|
||||
import static java.util.Collections.emptyList;
|
||||
import static java.util.Collections.emptyMap;
|
||||
import static java.util.stream.Collectors.toList;
|
||||
|
||||
public abstract class Expressions {
|
||||
|
@ -38,6 +39,18 @@ public abstract class Expressions {
|
|||
return list;
|
||||
}
|
||||
|
||||
public static AttributeMap<Expression> asAttributeMap(List<? extends NamedExpression> named) {
|
||||
if (named.isEmpty()) {
|
||||
return new AttributeMap<>(emptyMap());
|
||||
}
|
||||
|
||||
AttributeMap<Expression> map = new AttributeMap<>();
|
||||
for (NamedExpression exp : named) {
|
||||
map.add(exp.toAttribute(), exp);
|
||||
}
|
||||
return map;
|
||||
}
|
||||
|
||||
public static boolean anyMatch(List<? extends Expression> exps, Predicate<? super Expression> predicate) {
|
||||
for (Expression exp : exps) {
|
||||
if (exp.anyMatch(predicate)) {
|
||||
|
|
|
@ -7,7 +7,6 @@ package org.elasticsearch.xpack.sql.expression.function.aggregate;
|
|||
|
||||
import org.elasticsearch.xpack.sql.expression.Expression;
|
||||
import org.elasticsearch.xpack.sql.expression.function.Function;
|
||||
import org.elasticsearch.xpack.sql.querydsl.agg.AggPath;
|
||||
import org.elasticsearch.xpack.sql.tree.Location;
|
||||
import org.elasticsearch.xpack.sql.tree.NodeInfo;
|
||||
import org.elasticsearch.xpack.sql.type.DataType;
|
||||
|
@ -78,7 +77,12 @@ public class InnerAggregate extends AggregateFunction {
|
|||
public AggregateFunctionAttribute toAttribute() {
|
||||
// this is highly correlated with QueryFolder$FoldAggregate#addFunction (regarding the function name within the querydsl)
|
||||
return new AggregateFunctionAttribute(location(), name(), dataType(), outer.id(), functionId(),
|
||||
AggPath.metricValue(functionId(), innerId));
|
||||
aggMetricValue(functionId(), innerId));
|
||||
}
|
||||
|
||||
public static String aggMetricValue(String aggPath, String valueName) {
|
||||
// handle aggPath inconsistency (for percentiles and percentileRanks) percentile[99.9] (valid) vs percentile.99.9 (invalid)
|
||||
return aggPath + "[" + valueName + "]";
|
||||
}
|
||||
|
||||
@Override
|
||||
|
|
|
@ -11,7 +11,6 @@ import org.elasticsearch.xpack.sql.expression.Foldables;
|
|||
import org.elasticsearch.xpack.sql.tree.Location;
|
||||
import org.elasticsearch.xpack.sql.tree.NodeInfo;
|
||||
import org.elasticsearch.xpack.sql.type.DataType;
|
||||
import org.elasticsearch.xpack.sql.type.DataTypes;
|
||||
|
||||
import java.util.List;
|
||||
|
||||
|
@ -61,6 +60,6 @@ public class Percentile extends NumericAggregate implements EnclosedAgg {
|
|||
|
||||
@Override
|
||||
public String innerName() {
|
||||
return "[" + Double.toString(Foldables.doubleValueOf(percent)) + "]";
|
||||
return Double.toString(Foldables.doubleValueOf(percent));
|
||||
}
|
||||
}
|
||||
|
|
|
@ -11,7 +11,6 @@ import org.elasticsearch.xpack.sql.expression.Foldables;
|
|||
import org.elasticsearch.xpack.sql.tree.Location;
|
||||
import org.elasticsearch.xpack.sql.tree.NodeInfo;
|
||||
import org.elasticsearch.xpack.sql.type.DataType;
|
||||
import org.elasticsearch.xpack.sql.type.DataTypes;
|
||||
|
||||
import java.util.List;
|
||||
|
||||
|
@ -61,6 +60,6 @@ public class PercentileRank extends AggregateFunction implements EnclosedAgg {
|
|||
|
||||
@Override
|
||||
public String innerName() {
|
||||
return "[" + Double.toString(Foldables.doubleValueOf(value)) + "]";
|
||||
return Double.toString(Foldables.doubleValueOf(value));
|
||||
}
|
||||
}
|
||||
|
|
|
@ -11,16 +11,18 @@ import org.elasticsearch.xpack.sql.expression.function.scalar.arithmetic.BinaryA
|
|||
import org.elasticsearch.xpack.sql.expression.function.scalar.arithmetic.UnaryArithmeticProcessor;
|
||||
import org.elasticsearch.xpack.sql.expression.function.scalar.datetime.DateTimeProcessor;
|
||||
import org.elasticsearch.xpack.sql.expression.function.scalar.math.MathProcessor;
|
||||
import org.elasticsearch.xpack.sql.expression.function.scalar.processor.runtime.BucketExtractorProcessor;
|
||||
import org.elasticsearch.xpack.sql.expression.function.scalar.processor.runtime.ChainingProcessor;
|
||||
import org.elasticsearch.xpack.sql.expression.function.scalar.processor.runtime.ConstantProcessor;
|
||||
import org.elasticsearch.xpack.sql.expression.function.scalar.processor.runtime.HitExtractorProcessor;
|
||||
import org.elasticsearch.xpack.sql.expression.function.scalar.processor.runtime.MatrixFieldProcessor;
|
||||
import org.elasticsearch.xpack.sql.expression.function.scalar.processor.runtime.Processor;
|
||||
|
||||
import java.util.ArrayList;
|
||||
import java.util.List;
|
||||
|
||||
public abstract class Processors {
|
||||
public final class Processors {
|
||||
|
||||
private Processors() {}
|
||||
|
||||
/**
|
||||
* All of the named writeables needed to deserialize the instances of
|
||||
|
@ -31,9 +33,9 @@ public abstract class Processors {
|
|||
// base
|
||||
entries.add(new Entry(Processor.class, ConstantProcessor.NAME, ConstantProcessor::new));
|
||||
entries.add(new Entry(Processor.class, HitExtractorProcessor.NAME, HitExtractorProcessor::new));
|
||||
entries.add(new Entry(Processor.class, BucketExtractorProcessor.NAME, BucketExtractorProcessor::new));
|
||||
entries.add(new Entry(Processor.class, CastProcessor.NAME, CastProcessor::new));
|
||||
entries.add(new Entry(Processor.class, ChainingProcessor.NAME, ChainingProcessor::new));
|
||||
entries.add(new Entry(Processor.class, MatrixFieldProcessor.NAME, MatrixFieldProcessor::new));
|
||||
|
||||
// arithmetic
|
||||
entries.add(new Entry(Processor.class, BinaryArithmeticProcessor.NAME, BinaryArithmeticProcessor::new));
|
||||
|
|
|
@ -0,0 +1,51 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License;
|
||||
* you may not use this file except in compliance with the Elastic License.
|
||||
*/
|
||||
package org.elasticsearch.xpack.sql.expression.function.scalar.processor.definition;
|
||||
|
||||
import org.elasticsearch.xpack.sql.execution.search.SqlSourceBuilder;
|
||||
import org.elasticsearch.xpack.sql.execution.search.extractor.BucketExtractor;
|
||||
import org.elasticsearch.xpack.sql.expression.Expression;
|
||||
import org.elasticsearch.xpack.sql.expression.function.scalar.processor.runtime.BucketExtractorProcessor;
|
||||
import org.elasticsearch.xpack.sql.expression.function.scalar.processor.runtime.ChainingProcessor;
|
||||
import org.elasticsearch.xpack.sql.expression.function.scalar.processor.runtime.Processor;
|
||||
import org.elasticsearch.xpack.sql.tree.Location;
|
||||
import org.elasticsearch.xpack.sql.tree.NodeInfo;
|
||||
|
||||
public class AggExtractorInput extends LeafInput<BucketExtractor> {
|
||||
|
||||
private final Processor chained;
|
||||
|
||||
public AggExtractorInput(Location location, Expression expression, Processor processor, BucketExtractor context) {
|
||||
super(location, expression, context);
|
||||
this.chained = processor;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected NodeInfo<AggExtractorInput> info() {
|
||||
return NodeInfo.create(this, AggExtractorInput::new, expression(), chained, context());
|
||||
}
|
||||
|
||||
@Override
|
||||
public Processor asProcessor() {
|
||||
Processor proc = new BucketExtractorProcessor(context());
|
||||
return chained != null ? new ChainingProcessor(proc, chained) : proc;
|
||||
}
|
||||
|
||||
@Override
|
||||
public final boolean supportedByAggsOnlyQuery() {
|
||||
return true;
|
||||
}
|
||||
|
||||
@Override
|
||||
public ProcessorDefinition resolveAttributes(AttributeResolver resolver) {
|
||||
return this;
|
||||
}
|
||||
|
||||
@Override
|
||||
public final void collectFields(SqlSourceBuilder sourceBuilder) {
|
||||
// Nothing to collect
|
||||
}
|
||||
}
|
|
@ -5,39 +5,37 @@
|
|||
*/
|
||||
package org.elasticsearch.xpack.sql.expression.function.scalar.processor.definition;
|
||||
|
||||
import org.elasticsearch.xpack.sql.execution.search.AggRef;
|
||||
import org.elasticsearch.xpack.sql.expression.Expression;
|
||||
import org.elasticsearch.xpack.sql.expression.function.scalar.processor.runtime.Processor;
|
||||
import org.elasticsearch.xpack.sql.tree.Location;
|
||||
import org.elasticsearch.xpack.sql.tree.NodeInfo;
|
||||
|
||||
import java.util.Objects;
|
||||
|
||||
public class AggPathInput extends CommonNonExecutableInput<String> {
|
||||
public class AggPathInput extends CommonNonExecutableInput<AggRef> {
|
||||
|
||||
private final String innerKey;
|
||||
// used in case the agg itself is not returned in a suitable format (like date aggs)
|
||||
private final Processor action;
|
||||
|
||||
public AggPathInput(Expression expression, String context) {
|
||||
this(Location.EMPTY, expression, context, null, null);
|
||||
public AggPathInput(Expression expression, AggRef context) {
|
||||
this(Location.EMPTY, expression, context, null);
|
||||
}
|
||||
|
||||
public AggPathInput(Expression expression, String context, String innerKey) {
|
||||
this(Location.EMPTY, expression, context, innerKey, null);
|
||||
}
|
||||
|
||||
public AggPathInput(Location location, Expression expression, String context, String innerKey, Processor action) {
|
||||
/**
|
||||
*
|
||||
* Constructs a new <code>AggPathInput</code> instance.
|
||||
* The action is used for handling corner-case results such as date histogram which returns
|
||||
* a full date object for year which requires additional extraction.
|
||||
*/
|
||||
public AggPathInput(Location location, Expression expression, AggRef context, Processor action) {
|
||||
super(location, expression, context);
|
||||
this.innerKey = innerKey;
|
||||
this.action = action;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected NodeInfo<AggPathInput> info() {
|
||||
return NodeInfo.create(this, AggPathInput::new, expression(), context(), innerKey, action);
|
||||
}
|
||||
|
||||
public String innerKey() {
|
||||
return innerKey;
|
||||
return NodeInfo.create(this, AggPathInput::new, expression(), context(), action);
|
||||
}
|
||||
|
||||
public Processor action() {
|
||||
|
@ -56,7 +54,7 @@ public class AggPathInput extends CommonNonExecutableInput<String> {
|
|||
|
||||
@Override
|
||||
public int hashCode() {
|
||||
return Objects.hash(context(), innerKey);
|
||||
return Objects.hash(context(), action);
|
||||
}
|
||||
|
||||
@Override
|
||||
|
@ -71,7 +69,6 @@ public class AggPathInput extends CommonNonExecutableInput<String> {
|
|||
|
||||
AggPathInput other = (AggPathInput) obj;
|
||||
return Objects.equals(context(), other.context())
|
||||
&& Objects.equals(innerKey, other.innerKey)
|
||||
&& Objects.equals(action, other.action);
|
||||
}
|
||||
}
|
||||
|
|
|
@ -1,82 +0,0 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License;
|
||||
* you may not use this file except in compliance with the Elastic License.
|
||||
*/
|
||||
package org.elasticsearch.xpack.sql.expression.function.scalar.processor.definition;
|
||||
|
||||
import org.elasticsearch.xpack.sql.execution.search.SqlSourceBuilder;
|
||||
import org.elasticsearch.xpack.sql.expression.Expression;
|
||||
import org.elasticsearch.xpack.sql.expression.function.scalar.processor.runtime.MatrixFieldProcessor;
|
||||
import org.elasticsearch.xpack.sql.expression.function.scalar.processor.runtime.Processor;
|
||||
import org.elasticsearch.xpack.sql.expression.function.scalar.processor.runtime.SuppliedProcessor;
|
||||
import org.elasticsearch.xpack.sql.tree.Location;
|
||||
import org.elasticsearch.xpack.sql.tree.NodeInfo;
|
||||
import java.util.Objects;
|
||||
import java.util.function.Supplier;
|
||||
|
||||
public class AggValueInput extends LeafInput<Supplier<Object>> {
|
||||
|
||||
private final String innerKey;
|
||||
private final Processor matrixProcessor;
|
||||
|
||||
public AggValueInput(Location location, Expression expression, Supplier<Object> context, String innerKey) {
|
||||
super(location, expression, context);
|
||||
this.innerKey = innerKey;
|
||||
this.matrixProcessor = innerKey != null ? new MatrixFieldProcessor(innerKey) : null;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected NodeInfo<AggValueInput> info() {
|
||||
return NodeInfo.create(this, AggValueInput::new, expression(), context(), innerKey);
|
||||
}
|
||||
|
||||
public String innerKey() {
|
||||
return innerKey;
|
||||
}
|
||||
|
||||
@Override
|
||||
public Processor asProcessor() {
|
||||
return new SuppliedProcessor(() -> matrixProcessor != null ? matrixProcessor.process(context().get()) : context().get());
|
||||
}
|
||||
|
||||
@Override
|
||||
public final boolean supportedByAggsOnlyQuery() {
|
||||
return true;
|
||||
}
|
||||
|
||||
@Override
|
||||
public ProcessorDefinition resolveAttributes(AttributeResolver resolver) {
|
||||
return this;
|
||||
}
|
||||
|
||||
@Override
|
||||
public final void collectFields(SqlSourceBuilder sourceBuilder) {
|
||||
// Nothing to collect
|
||||
}
|
||||
|
||||
@Override
|
||||
public final int depth() {
|
||||
return 0;
|
||||
}
|
||||
|
||||
@Override
|
||||
public int hashCode() {
|
||||
return Objects.hash(context(), innerKey);
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean equals(Object obj) {
|
||||
if (this == obj) {
|
||||
return true;
|
||||
}
|
||||
|
||||
if (obj == null || getClass() != obj.getClass()) {
|
||||
return false;
|
||||
}
|
||||
|
||||
AggValueInput other = (AggValueInput) obj;
|
||||
return Objects.equals(context(), other.context())
|
||||
&& Objects.equals(innerKey, other.innerKey);
|
||||
}
|
||||
}
|
|
@ -39,9 +39,4 @@ public class AttributeInput extends NonExecutableInput<Attribute> {
|
|||
public final void collectFields(SqlSourceBuilder sourceBuilder) {
|
||||
// Nothing to extract
|
||||
}
|
||||
|
||||
@Override
|
||||
public final int depth() {
|
||||
return 0;
|
||||
}
|
||||
}
|
||||
|
|
|
@ -8,6 +8,7 @@ package org.elasticsearch.xpack.sql.expression.function.scalar.processor.definit
|
|||
import org.elasticsearch.xpack.sql.execution.search.SqlSourceBuilder;
|
||||
import org.elasticsearch.xpack.sql.expression.Expression;
|
||||
import org.elasticsearch.xpack.sql.tree.Location;
|
||||
|
||||
import java.util.Arrays;
|
||||
import java.util.List;
|
||||
|
||||
|
@ -52,11 +53,6 @@ public abstract class BinaryProcessorDefinition extends ProcessorDefinition {
|
|||
return replaceChildren(newLeft, newRight);
|
||||
}
|
||||
|
||||
@Override
|
||||
public final int depth() {
|
||||
return Math.max(left.depth(), right.depth());
|
||||
}
|
||||
|
||||
/**
|
||||
* Build a copy of this object with new left and right children. Used by
|
||||
* {@link #resolveAttributes(AttributeResolver)}.
|
||||
|
|
|
@ -34,9 +34,4 @@ abstract class CommonNonExecutableInput<T> extends NonExecutableInput<T> {
|
|||
public final void collectFields(SqlSourceBuilder sourceBuilder) {
|
||||
// Nothing to extract
|
||||
}
|
||||
|
||||
@Override
|
||||
public final int depth() {
|
||||
return 0;
|
||||
}
|
||||
}
|
||||
|
|
|
@ -42,9 +42,4 @@ public class ConstantInput extends LeafInput<Object> {
|
|||
public final void collectFields(SqlSourceBuilder sourceBuilder) {
|
||||
// Nothing to collect
|
||||
}
|
||||
|
||||
@Override
|
||||
public final int depth() {
|
||||
return 0;
|
||||
}
|
||||
}
|
||||
|
|
|
@ -43,9 +43,4 @@ public class HitExtractorInput extends LeafInput<HitExtractor> {
|
|||
public final void collectFields(SqlSourceBuilder sourceBuilder) {
|
||||
// No fields to collect
|
||||
}
|
||||
|
||||
@Override
|
||||
public final int depth() {
|
||||
return 0;
|
||||
}
|
||||
}
|
||||
|
|
|
@ -7,12 +7,12 @@ package org.elasticsearch.xpack.sql.expression.function.scalar.processor.definit
|
|||
|
||||
import org.elasticsearch.xpack.sql.expression.Expression;
|
||||
import org.elasticsearch.xpack.sql.tree.Location;
|
||||
|
||||
import java.util.List;
|
||||
import java.util.Objects;
|
||||
|
||||
import static java.util.Collections.emptyList;
|
||||
|
||||
import java.util.List;
|
||||
|
||||
public abstract class LeafInput<T> extends ProcessorDefinition {
|
||||
|
||||
private T context;
|
||||
|
|
|
@ -35,9 +35,4 @@ public class ReferenceInput extends NonExecutableInput<FieldExtraction> {
|
|||
public final void collectFields(SqlSourceBuilder sourceBuilder) {
|
||||
context().collectFields(sourceBuilder);
|
||||
}
|
||||
|
||||
@Override
|
||||
public int depth() {
|
||||
return context().depth();
|
||||
}
|
||||
}
|
||||
|
|
|
@ -8,15 +8,15 @@ package org.elasticsearch.xpack.sql.expression.function.scalar.processor.definit
|
|||
import org.elasticsearch.xpack.sql.execution.search.SqlSourceBuilder;
|
||||
import org.elasticsearch.xpack.sql.execution.search.extractor.ScoreExtractor;
|
||||
import org.elasticsearch.xpack.sql.expression.Expression;
|
||||
import org.elasticsearch.xpack.sql.expression.function.scalar.processor.runtime.HitExtractorProcessor;
|
||||
import org.elasticsearch.xpack.sql.expression.function.scalar.processor.runtime.Processor;
|
||||
import org.elasticsearch.xpack.sql.tree.Location;
|
||||
import org.elasticsearch.xpack.sql.tree.NodeInfo;
|
||||
import org.elasticsearch.xpack.sql.expression.function.scalar.processor.runtime.HitExtractorProcessor;
|
||||
|
||||
import static java.util.Collections.emptyList;
|
||||
|
||||
import java.util.List;
|
||||
|
||||
import static java.util.Collections.emptyList;
|
||||
|
||||
public class ScoreProcessorDefinition extends ProcessorDefinition {
|
||||
public ScoreProcessorDefinition(Location location, Expression expression) {
|
||||
super(location, expression, emptyList());
|
||||
|
@ -56,9 +56,4 @@ public class ScoreProcessorDefinition extends ProcessorDefinition {
|
|||
public void collectFields(SqlSourceBuilder sourceBuilder) {
|
||||
sourceBuilder.trackScores();
|
||||
}
|
||||
|
||||
@Override
|
||||
public int depth() {
|
||||
return 0;
|
||||
}
|
||||
}
|
||||
|
|
|
@ -11,12 +11,12 @@ import org.elasticsearch.xpack.sql.expression.function.scalar.processor.runtime.
|
|||
import org.elasticsearch.xpack.sql.expression.function.scalar.processor.runtime.Processor;
|
||||
import org.elasticsearch.xpack.sql.tree.Location;
|
||||
import org.elasticsearch.xpack.sql.tree.NodeInfo;
|
||||
|
||||
import java.util.List;
|
||||
import java.util.Objects;
|
||||
|
||||
import static java.util.Collections.singletonList;
|
||||
|
||||
import java.util.List;
|
||||
|
||||
public final class UnaryProcessorDefinition extends ProcessorDefinition {
|
||||
|
||||
private final ProcessorDefinition child;
|
||||
|
@ -78,11 +78,6 @@ public final class UnaryProcessorDefinition extends ProcessorDefinition {
|
|||
child.collectFields(sourceBuilder);
|
||||
}
|
||||
|
||||
@Override
|
||||
public int depth() {
|
||||
return child.depth();
|
||||
}
|
||||
|
||||
@Override
|
||||
public int hashCode() {
|
||||
return Objects.hash(expression(), child, action);
|
||||
|
|
|
@ -0,0 +1,76 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License;
|
||||
* you may not use this file except in compliance with the Elastic License.
|
||||
*/
|
||||
package org.elasticsearch.xpack.sql.expression.function.scalar.processor.runtime;
|
||||
|
||||
import org.elasticsearch.common.io.stream.StreamInput;
|
||||
import org.elasticsearch.common.io.stream.StreamOutput;
|
||||
import org.elasticsearch.search.aggregations.bucket.MultiBucketsAggregation.Bucket;
|
||||
import org.elasticsearch.xpack.sql.SqlIllegalArgumentException;
|
||||
import org.elasticsearch.xpack.sql.execution.search.extractor.BucketExtractor;
|
||||
|
||||
import java.io.IOException;
|
||||
import java.util.Objects;
|
||||
|
||||
/**
|
||||
* Processor wrapping an {@link BucketExtractor}, essentially being a source/leaf of a
|
||||
* Processor tree.
|
||||
*/
|
||||
public class BucketExtractorProcessor implements Processor {
|
||||
|
||||
public static final String NAME = "a";
|
||||
|
||||
private final BucketExtractor extractor;
|
||||
|
||||
public BucketExtractorProcessor(BucketExtractor extractor) {
|
||||
this.extractor = extractor;
|
||||
}
|
||||
|
||||
public BucketExtractorProcessor(StreamInput in) throws IOException {
|
||||
extractor = in.readNamedWriteable(BucketExtractor.class);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void writeTo(StreamOutput out) throws IOException {
|
||||
out.writeNamedWriteable(extractor);
|
||||
}
|
||||
|
||||
@Override
|
||||
public String getWriteableName() {
|
||||
return NAME;
|
||||
}
|
||||
|
||||
@Override
|
||||
public Object process(Object input) {
|
||||
if (!(input instanceof Bucket)) {
|
||||
throw new SqlIllegalArgumentException("Expected an agg bucket but received {}", input);
|
||||
}
|
||||
return extractor.extract((Bucket) input);
|
||||
}
|
||||
|
||||
@Override
|
||||
public int hashCode() {
|
||||
return Objects.hash(extractor);
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean equals(Object obj) {
|
||||
if (this == obj) {
|
||||
return true;
|
||||
}
|
||||
|
||||
if (obj == null || getClass() != obj.getClass()) {
|
||||
return false;
|
||||
}
|
||||
|
||||
BucketExtractorProcessor other = (BucketExtractorProcessor) obj;
|
||||
return Objects.equals(extractor, other.extractor);
|
||||
}
|
||||
|
||||
@Override
|
||||
public String toString() {
|
||||
return extractor.toString();
|
||||
}
|
||||
}
|
|
@ -15,7 +15,7 @@ import java.io.IOException;
|
|||
import java.util.Objects;
|
||||
|
||||
/**
|
||||
* Processor wrapping a HitExtractor esentially being a source/leaf of a
|
||||
* Processor wrapping a {@link HitExtractor}, essentially being a source/leaf of a
|
||||
* Processor tree.
|
||||
*/
|
||||
public class HitExtractorProcessor implements Processor {
|
||||
|
@ -47,7 +47,7 @@ public class HitExtractorProcessor implements Processor {
|
|||
if (!(input instanceof SearchHit)) {
|
||||
throw new SqlIllegalArgumentException("Expected a SearchHit but received {}", input);
|
||||
}
|
||||
return extractor.get((SearchHit) input);
|
||||
return extractor.extract((SearchHit) input);
|
||||
}
|
||||
|
||||
@Override
|
||||
|
|
|
@ -1,69 +0,0 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License;
|
||||
* you may not use this file except in compliance with the Elastic License.
|
||||
*/
|
||||
package org.elasticsearch.xpack.sql.expression.function.scalar.processor.runtime;
|
||||
|
||||
import org.elasticsearch.common.io.stream.StreamInput;
|
||||
import org.elasticsearch.common.io.stream.StreamOutput;
|
||||
|
||||
import java.io.IOException;
|
||||
import java.util.Map;
|
||||
import java.util.Objects;
|
||||
|
||||
public class MatrixFieldProcessor implements Processor {
|
||||
public static final String NAME = "mat";
|
||||
|
||||
private final String key;
|
||||
|
||||
public MatrixFieldProcessor(String key) {
|
||||
this.key = key;
|
||||
}
|
||||
|
||||
public MatrixFieldProcessor(StreamInput in) throws IOException {
|
||||
key = in.readString();
|
||||
}
|
||||
|
||||
@Override
|
||||
public void writeTo(StreamOutput out) throws IOException {
|
||||
out.writeString(key);
|
||||
}
|
||||
|
||||
@Override
|
||||
public String getWriteableName() {
|
||||
return NAME;
|
||||
}
|
||||
|
||||
String key() {
|
||||
return key;
|
||||
}
|
||||
|
||||
@Override
|
||||
public Object process(Object r) {
|
||||
return r instanceof Map ? ((Map<?, ?>) r).get(key) : r;
|
||||
}
|
||||
|
||||
@Override
|
||||
public int hashCode() {
|
||||
return Objects.hashCode(key);
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean equals(Object obj) {
|
||||
if (this == obj) {
|
||||
return true;
|
||||
}
|
||||
|
||||
if (obj == null || getClass() != obj.getClass()) {
|
||||
return false;
|
||||
}
|
||||
|
||||
MatrixFieldProcessor other = (MatrixFieldProcessor) obj;
|
||||
return Objects.equals(key, other.key);
|
||||
}
|
||||
|
||||
public String toString() {
|
||||
return "[" + key + "]";
|
||||
}
|
||||
}
|
|
@ -1,37 +0,0 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License;
|
||||
* you may not use this file except in compliance with the Elastic License.
|
||||
*/
|
||||
package org.elasticsearch.xpack.sql.expression.function.scalar.processor.runtime;
|
||||
|
||||
import org.elasticsearch.common.io.stream.StreamOutput;
|
||||
import org.elasticsearch.xpack.sql.SqlIllegalArgumentException;
|
||||
|
||||
import java.io.IOException;
|
||||
import java.util.function.Supplier;
|
||||
|
||||
public class SuppliedProcessor implements Processor {
|
||||
|
||||
private final Supplier<Object> action;
|
||||
|
||||
public SuppliedProcessor(Supplier<Object> action) {
|
||||
this.action = action;
|
||||
}
|
||||
|
||||
@Override
|
||||
public String getWriteableName() {
|
||||
throw new SqlIllegalArgumentException("transient");
|
||||
}
|
||||
|
||||
@Override
|
||||
public void writeTo(StreamOutput out) throws IOException {
|
||||
throw new SqlIllegalArgumentException("transient");
|
||||
}
|
||||
|
||||
@Override
|
||||
public Object process(Object input) {
|
||||
return action.get();
|
||||
}
|
||||
|
||||
}
|
|
@ -6,7 +6,7 @@
|
|||
package org.elasticsearch.xpack.sql.plan.physical;
|
||||
|
||||
import org.elasticsearch.action.ActionListener;
|
||||
import org.elasticsearch.xpack.sql.execution.search.Scroller;
|
||||
import org.elasticsearch.xpack.sql.execution.search.Querier;
|
||||
import org.elasticsearch.xpack.sql.expression.Attribute;
|
||||
import org.elasticsearch.xpack.sql.querydsl.container.QueryContainer;
|
||||
import org.elasticsearch.xpack.sql.session.Rows;
|
||||
|
@ -56,8 +56,8 @@ public class EsQueryExec extends LeafExec {
|
|||
|
||||
@Override
|
||||
public void execute(SqlSession session, ActionListener<SchemaRowSet> listener) {
|
||||
Scroller scroller = new Scroller(session.client(), session.settings());
|
||||
scroller.scroll(Rows.schema(output), queryContainer, index, listener);
|
||||
Querier scroller = new Querier(session.client(), session.settings());
|
||||
scroller.query(Rows.schema(output), queryContainer, index, listener);
|
||||
}
|
||||
|
||||
@Override
|
||||
|
|
|
@ -6,6 +6,7 @@
|
|||
package org.elasticsearch.xpack.sql.planner;
|
||||
|
||||
import org.elasticsearch.common.collect.Tuple;
|
||||
import org.elasticsearch.xpack.sql.execution.search.AggRef;
|
||||
import org.elasticsearch.xpack.sql.expression.Alias;
|
||||
import org.elasticsearch.xpack.sql.expression.Attribute;
|
||||
import org.elasticsearch.xpack.sql.expression.Expression;
|
||||
|
@ -22,6 +23,7 @@ import org.elasticsearch.xpack.sql.expression.function.aggregate.Count;
|
|||
import org.elasticsearch.xpack.sql.expression.function.aggregate.InnerAggregate;
|
||||
import org.elasticsearch.xpack.sql.expression.function.scalar.ScalarFunction;
|
||||
import org.elasticsearch.xpack.sql.expression.function.scalar.ScalarFunctionAttribute;
|
||||
import org.elasticsearch.xpack.sql.expression.function.scalar.datetime.DateTimeFunction;
|
||||
import org.elasticsearch.xpack.sql.expression.function.scalar.datetime.DateTimeHistogramFunction;
|
||||
import org.elasticsearch.xpack.sql.expression.function.scalar.processor.definition.AggPathInput;
|
||||
import org.elasticsearch.xpack.sql.expression.function.scalar.processor.definition.ProcessorDefinition;
|
||||
|
@ -39,23 +41,26 @@ import org.elasticsearch.xpack.sql.plan.physical.ProjectExec;
|
|||
import org.elasticsearch.xpack.sql.planner.QueryTranslator.GroupingContext;
|
||||
import org.elasticsearch.xpack.sql.planner.QueryTranslator.QueryTranslation;
|
||||
import org.elasticsearch.xpack.sql.querydsl.agg.AggFilter;
|
||||
import org.elasticsearch.xpack.sql.querydsl.agg.AggPath;
|
||||
import org.elasticsearch.xpack.sql.querydsl.agg.Aggs;
|
||||
import org.elasticsearch.xpack.sql.querydsl.agg.GroupingAgg;
|
||||
import org.elasticsearch.xpack.sql.querydsl.agg.GroupByKey;
|
||||
import org.elasticsearch.xpack.sql.querydsl.agg.LeafAgg;
|
||||
import org.elasticsearch.xpack.sql.querydsl.container.AttributeSort;
|
||||
import org.elasticsearch.xpack.sql.querydsl.container.ComputedRef;
|
||||
import org.elasticsearch.xpack.sql.querydsl.container.GlobalCountRef;
|
||||
import org.elasticsearch.xpack.sql.querydsl.container.GroupByRef;
|
||||
import org.elasticsearch.xpack.sql.querydsl.container.GroupByRef.Property;
|
||||
import org.elasticsearch.xpack.sql.querydsl.container.MetricAggRef;
|
||||
import org.elasticsearch.xpack.sql.querydsl.container.QueryContainer;
|
||||
import org.elasticsearch.xpack.sql.querydsl.container.ScoreSort;
|
||||
import org.elasticsearch.xpack.sql.querydsl.container.ScriptSort;
|
||||
import org.elasticsearch.xpack.sql.querydsl.container.Sort.Direction;
|
||||
import org.elasticsearch.xpack.sql.querydsl.container.TotalCountRef;
|
||||
import org.elasticsearch.xpack.sql.querydsl.query.Query;
|
||||
import org.elasticsearch.xpack.sql.rule.Rule;
|
||||
import org.elasticsearch.xpack.sql.rule.RuleExecutor;
|
||||
import org.elasticsearch.xpack.sql.session.EmptyExecutable;
|
||||
import org.elasticsearch.xpack.sql.type.DataType;
|
||||
import org.elasticsearch.xpack.sql.util.Check;
|
||||
import org.elasticsearch.xpack.sql.util.StringUtils;
|
||||
import org.joda.time.DateTimeZone;
|
||||
|
||||
import java.util.Arrays;
|
||||
import java.util.LinkedHashMap;
|
||||
|
@ -65,8 +70,10 @@ import java.util.concurrent.atomic.AtomicReference;
|
|||
import static org.elasticsearch.xpack.sql.planner.QueryTranslator.and;
|
||||
import static org.elasticsearch.xpack.sql.planner.QueryTranslator.toAgg;
|
||||
import static org.elasticsearch.xpack.sql.planner.QueryTranslator.toQuery;
|
||||
import static org.elasticsearch.xpack.sql.util.CollectionUtils.combine;
|
||||
|
||||
/**
|
||||
* Folds the PhysicalPlan into a {@link Query}.
|
||||
*/
|
||||
class QueryFolder extends RuleExecutor<PhysicalPlan> {
|
||||
PhysicalPlan fold(PhysicalPlan plan) {
|
||||
return execute(plan);
|
||||
|
@ -175,51 +182,8 @@ class QueryFolder extends RuleExecutor<PhysicalPlan> {
|
|||
if (filter == null) {
|
||||
return qContainer.aggs();
|
||||
}
|
||||
|
||||
// find the relevant groups and compute the shortest path (the highest group in the hierarchy)
|
||||
Map<String, GroupingAgg> groupPaths = new LinkedHashMap<>();
|
||||
// root group
|
||||
String shortestPath = null;
|
||||
GroupingAgg targetGroup = null;
|
||||
|
||||
for (String refId : filter.aggRefs()) {
|
||||
// is it root group or agg property (_count)
|
||||
if (refId == null) {
|
||||
shortestPath = StringUtils.EMPTY;
|
||||
}
|
||||
else {
|
||||
// find function group
|
||||
GroupingAgg groupAgg = qContainer.findGroupForAgg(refId);
|
||||
|
||||
if (groupAgg == null) {
|
||||
groupAgg = qContainer.pseudoFunctions().get(refId);
|
||||
}
|
||||
|
||||
if (groupAgg == null) {
|
||||
// Weird ctor call to make sure we don't interpret the message as a pattern
|
||||
throw new FoldingException(fexec, "Cannot find group for agg " + refId
|
||||
+ " referrenced by agg filter " + filter.name() + "(" + filter + ")", (Exception) null);
|
||||
}
|
||||
|
||||
String path = groupAgg.asParentPath();
|
||||
if (shortestPath == null || shortestPath.length() > path.length()) {
|
||||
shortestPath = path;
|
||||
targetGroup = groupAgg;
|
||||
}
|
||||
groupPaths.put(refId, groupAgg);
|
||||
}
|
||||
}
|
||||
|
||||
// and finally update the agg groups
|
||||
if (targetGroup == GroupingAgg.DEFAULT_GROUP) {
|
||||
throw new PlanningException("Aggregation filtering not supported (yet) without explicit grouping");
|
||||
//aggs = aggs.addAgg(null, filter);
|
||||
}
|
||||
if (targetGroup == null) {
|
||||
throw new PlanningException("Cannot determine group column; likely an invalid query - please report");
|
||||
}
|
||||
else {
|
||||
aggs = aggs.updateGroup(targetGroup.withPipelines(combine(targetGroup.subPipelines(), filter)));
|
||||
aggs = aggs.addAgg(filter);
|
||||
}
|
||||
|
||||
return aggs;
|
||||
|
@ -296,7 +260,7 @@ class QueryFolder extends RuleExecutor<PhysicalPlan> {
|
|||
// get the backing expression and check if it belongs to a agg group or whether it's
|
||||
// an expression in the first place
|
||||
Expression exp = p.expression();
|
||||
GroupingAgg matchingGroup = null;
|
||||
GroupByKey matchingGroup = null;
|
||||
if (groupingContext != null) {
|
||||
// is there a group (aggregation) for this expression ?
|
||||
matchingGroup = groupingContext.groupFor(exp);
|
||||
|
@ -315,6 +279,7 @@ class QueryFolder extends RuleExecutor<PhysicalPlan> {
|
|||
if (matchingGroup != null) {
|
||||
if (exp instanceof Attribute || exp instanceof ScalarFunction) {
|
||||
Processor action = null;
|
||||
DateTimeZone tz = null;
|
||||
/*
|
||||
* special handling of dates since aggs return the typed Date object which needs
|
||||
* extraction instead of handling this in the scroller, the folder handles this
|
||||
|
@ -322,8 +287,9 @@ class QueryFolder extends RuleExecutor<PhysicalPlan> {
|
|||
*/
|
||||
if (exp instanceof DateTimeHistogramFunction) {
|
||||
action = ((UnaryProcessorDefinition) p).action();
|
||||
tz = ((DateTimeFunction) exp).timeZone();
|
||||
}
|
||||
return new AggPathInput(exp.location(), exp, matchingGroup.propertyPath(), null, action);
|
||||
return new AggPathInput(exp.location(), exp, new GroupByRef(matchingGroup.id(), null, tz), action);
|
||||
}
|
||||
}
|
||||
// or found an aggregate expression (which has to work on an attribute used for grouping)
|
||||
|
@ -350,35 +316,41 @@ class QueryFolder extends RuleExecutor<PhysicalPlan> {
|
|||
// already used in the aggpath)
|
||||
//aliases.put(as.toAttribute(), sf.toAttribute());
|
||||
}
|
||||
// apply the same logic above (for function inputs) to non-scalar functions with small variantions:
|
||||
// apply the same logic above (for function inputs) to non-scalar functions with small variations:
|
||||
// instead of adding things as input, add them as full blown column
|
||||
else {
|
||||
GroupingAgg matchingGroup = null;
|
||||
GroupByKey matchingGroup = null;
|
||||
if (groupingContext != null) {
|
||||
// is there a group (aggregation) for this expression ?
|
||||
matchingGroup = groupingContext.groupFor(child);
|
||||
}
|
||||
// attributes can only refer to declared groups
|
||||
if (child instanceof Attribute) {
|
||||
Check.notNull(matchingGroup, "Cannot find group '%s'", Expressions.name(child));
|
||||
queryC = queryC.addAggColumn(matchingGroup.propertyPath());
|
||||
Check.notNull(matchingGroup, "Cannot find group [{}]", Expressions.name(child));
|
||||
// check if the field is a date - if so mark it as such to interpret the long as a date
|
||||
// UTC is used since that's what the server uses and there's no conversion applied
|
||||
// (like for date histograms)
|
||||
DateTimeZone dt = DataType.DATE == child.dataType() ? DateTimeZone.UTC : null;
|
||||
queryC = queryC.addColumn(new GroupByRef(matchingGroup.id(), null, dt));
|
||||
}
|
||||
else {
|
||||
// the only thing left is agg function
|
||||
Check.isTrue(Functions.isAggregate(child),
|
||||
"Expected aggregate function inside alias; got %s", child.nodeString());
|
||||
"Expected aggregate function inside alias; got [{}]", child.nodeString());
|
||||
Tuple<QueryContainer, AggPathInput> withAgg = addAggFunction(matchingGroup,
|
||||
(AggregateFunction) child, compoundAggMap, queryC);
|
||||
queryC = withAgg.v1().addAggColumn(withAgg.v2().context(), withAgg.v2().innerKey());
|
||||
queryC = withAgg.v1().addColumn(withAgg.v2().context());
|
||||
}
|
||||
}
|
||||
// not an Alias or Function means it's an Attribute so apply the same logic as above
|
||||
} else {
|
||||
GroupingAgg matchingGroup = null;
|
||||
GroupByKey matchingGroup = null;
|
||||
if (groupingContext != null) {
|
||||
matchingGroup = groupingContext.groupFor(ne);
|
||||
Check.notNull(matchingGroup, "Cannot find group '%s'", Expressions.name(ne));
|
||||
queryC = queryC.addAggColumn(matchingGroup.propertyPath());
|
||||
Check.notNull(matchingGroup, "Cannot find group [{}]", Expressions.name(ne));
|
||||
|
||||
DateTimeZone dt = DataType.DATE == ne.dataType() ? DateTimeZone.UTC : null;
|
||||
queryC = queryC.addColumn(new GroupByRef(matchingGroup.id(), null, dt));
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -393,26 +365,25 @@ class QueryFolder extends RuleExecutor<PhysicalPlan> {
|
|||
return a;
|
||||
}
|
||||
|
||||
private Tuple<QueryContainer, AggPathInput> addAggFunction(GroupingAgg parentAgg, AggregateFunction f,
|
||||
private Tuple<QueryContainer, AggPathInput> addAggFunction(GroupByKey groupingAgg, AggregateFunction f,
|
||||
Map<CompoundNumericAggregate, String> compoundAggMap, QueryContainer queryC) {
|
||||
String functionId = f.functionId();
|
||||
// handle count as a special case agg
|
||||
if (f instanceof Count) {
|
||||
Count c = (Count) f;
|
||||
if (!c.distinct()) {
|
||||
String path = parentAgg == null ? TotalCountRef.PATH : AggPath.bucketCount(parentAgg.asParentPath());
|
||||
Map<String, GroupingAgg> pseudoFunctions = new LinkedHashMap<>(queryC.pseudoFunctions());
|
||||
pseudoFunctions.put(functionId, parentAgg);
|
||||
return new Tuple<>(queryC.withPseudoFunctions(pseudoFunctions), new AggPathInput(f, path));
|
||||
AggRef ref = groupingAgg == null ?
|
||||
GlobalCountRef.INSTANCE :
|
||||
new GroupByRef(groupingAgg.id(), Property.COUNT, null);
|
||||
|
||||
Map<String, GroupByKey> pseudoFunctions = new LinkedHashMap<>(queryC.pseudoFunctions());
|
||||
pseudoFunctions.put(functionId, groupingAgg);
|
||||
return new Tuple<>(queryC.withPseudoFunctions(pseudoFunctions), new AggPathInput(f, ref));
|
||||
}
|
||||
}
|
||||
|
||||
AggPathInput aggInput = null;
|
||||
|
||||
// otherwise translate the function into an actual agg
|
||||
String parentPath = parentAgg != null ? parentAgg.asParentPath() : null;
|
||||
String groupId = parentAgg != null ? parentAgg.id() : null;
|
||||
|
||||
if (f instanceof InnerAggregate) {
|
||||
InnerAggregate ia = (InnerAggregate) f;
|
||||
CompoundNumericAggregate outer = ia.outer();
|
||||
|
@ -420,22 +391,22 @@ class QueryFolder extends RuleExecutor<PhysicalPlan> {
|
|||
|
||||
// the compound agg hasn't been seen before so initialize it
|
||||
if (cAggPath == null) {
|
||||
LeafAgg leafAgg = toAgg(parentPath, functionId, outer);
|
||||
cAggPath = leafAgg.propertyPath();
|
||||
LeafAgg leafAgg = toAgg(outer.functionId(), outer);
|
||||
cAggPath = leafAgg.id();
|
||||
compoundAggMap.put(outer, cAggPath);
|
||||
// add the agg (without any reference)
|
||||
queryC = queryC.with(queryC.aggs().addAgg(leafAgg));
|
||||
}
|
||||
|
||||
String aggPath = AggPath.metricValue(cAggPath, ia.innerId());
|
||||
// FIXME: concern leak - hack around MatrixAgg which is not
|
||||
// generalized (afaik)
|
||||
aggInput = new AggPathInput(f, aggPath, ia.innerKey() != null ? QueryTranslator.nameOf(ia.innerKey()) : null);
|
||||
aggInput = new AggPathInput(f,
|
||||
new MetricAggRef(cAggPath, ia.innerId(), ia.innerKey() != null ? QueryTranslator.nameOf(ia.innerKey()) : null));
|
||||
}
|
||||
else {
|
||||
LeafAgg leafAgg = toAgg(parentPath, functionId, f);
|
||||
aggInput = new AggPathInput(f, leafAgg.propertyPath());
|
||||
queryC = queryC.with(queryC.aggs().addAgg(groupId, leafAgg));
|
||||
LeafAgg leafAgg = toAgg(functionId, f);
|
||||
aggInput = new AggPathInput(f, new MetricAggRef(leafAgg.id()));
|
||||
queryC = queryC.with(queryC.aggs().addAgg(leafAgg));
|
||||
}
|
||||
|
||||
return new Tuple<>(queryC, aggInput);
|
||||
|
@ -457,17 +428,17 @@ class QueryFolder extends RuleExecutor<PhysicalPlan> {
|
|||
// check whether there's an alias (occurs with scalar functions which are not named)
|
||||
attr = qContainer.aliases().getOrDefault(attr, attr);
|
||||
String lookup = attr.id().toString();
|
||||
GroupingAgg group = qContainer.findGroupForAgg(lookup);
|
||||
GroupByKey group = qContainer.findGroupForAgg(lookup);
|
||||
|
||||
// TODO: might need to validate whether the target field or group actually exist
|
||||
if (group != null && group != GroupingAgg.DEFAULT_GROUP) {
|
||||
if (group != null && group != Aggs.IMPLICIT_GROUP_KEY) {
|
||||
// check whether the lookup matches a group
|
||||
if (group.id().equals(lookup)) {
|
||||
qContainer = qContainer.updateGroup(group.with(direction));
|
||||
}
|
||||
// else it's a leafAgg
|
||||
else {
|
||||
qContainer = qContainer.updateGroup(group.with(lookup, direction));
|
||||
qContainer = qContainer.updateGroup(group.with(direction));
|
||||
}
|
||||
}
|
||||
else {
|
||||
|
|
|
@ -53,15 +53,14 @@ import org.elasticsearch.xpack.sql.expression.regex.Like;
|
|||
import org.elasticsearch.xpack.sql.expression.regex.LikePattern;
|
||||
import org.elasticsearch.xpack.sql.expression.regex.RLike;
|
||||
import org.elasticsearch.xpack.sql.querydsl.agg.AggFilter;
|
||||
import org.elasticsearch.xpack.sql.querydsl.agg.AggPath;
|
||||
import org.elasticsearch.xpack.sql.querydsl.agg.AndAggFilter;
|
||||
import org.elasticsearch.xpack.sql.querydsl.agg.AvgAgg;
|
||||
import org.elasticsearch.xpack.sql.querydsl.agg.CardinalityAgg;
|
||||
import org.elasticsearch.xpack.sql.querydsl.agg.ExtendedStatsAgg;
|
||||
import org.elasticsearch.xpack.sql.querydsl.agg.GroupByColumnAgg;
|
||||
import org.elasticsearch.xpack.sql.querydsl.agg.GroupByDateAgg;
|
||||
import org.elasticsearch.xpack.sql.querydsl.agg.GroupByScriptAgg;
|
||||
import org.elasticsearch.xpack.sql.querydsl.agg.GroupingAgg;
|
||||
import org.elasticsearch.xpack.sql.querydsl.agg.GroupByColumnKey;
|
||||
import org.elasticsearch.xpack.sql.querydsl.agg.GroupByDateKey;
|
||||
import org.elasticsearch.xpack.sql.querydsl.agg.GroupByKey;
|
||||
import org.elasticsearch.xpack.sql.querydsl.agg.GroupByScriptKey;
|
||||
import org.elasticsearch.xpack.sql.querydsl.agg.LeafAgg;
|
||||
import org.elasticsearch.xpack.sql.querydsl.agg.MatrixStatsAgg;
|
||||
import org.elasticsearch.xpack.sql.querydsl.agg.MaxAgg;
|
||||
|
@ -89,9 +88,7 @@ import org.elasticsearch.xpack.sql.type.DataType;
|
|||
import org.elasticsearch.xpack.sql.util.Check;
|
||||
import org.elasticsearch.xpack.sql.util.ReflectionUtils;
|
||||
|
||||
import java.util.ArrayList;
|
||||
import java.util.Arrays;
|
||||
import java.util.Iterator;
|
||||
import java.util.LinkedHashMap;
|
||||
import java.util.List;
|
||||
import java.util.Locale;
|
||||
|
@ -165,10 +162,10 @@ abstract class QueryTranslator {
|
|||
throw new UnsupportedOperationException(format(Locale.ROOT, "Don't know how to translate %s %s", e.nodeName(), e));
|
||||
}
|
||||
|
||||
static LeafAgg toAgg(String parent, String id, Function f) {
|
||||
static LeafAgg toAgg(String id, Function f) {
|
||||
|
||||
for (AggTranslator<?> translator : AGG_TRANSLATORS) {
|
||||
LeafAgg agg = translator.apply(id, parent, f);
|
||||
LeafAgg agg = translator.apply(id, f);
|
||||
if (agg != null) {
|
||||
return agg;
|
||||
}
|
||||
|
@ -178,44 +175,26 @@ abstract class QueryTranslator {
|
|||
}
|
||||
|
||||
static class GroupingContext {
|
||||
final GroupingAgg head;
|
||||
final GroupingAgg tail;
|
||||
final ExpressionId headAggId;
|
||||
final Map<ExpressionId, GroupByKey> groupMap;
|
||||
final GroupByKey tail;
|
||||
|
||||
final Map<ExpressionId, GroupingAgg> groupMap;
|
||||
final List<String> aggNames;
|
||||
final String groupPath;
|
||||
|
||||
GroupingContext(Map<ExpressionId, GroupingAgg> groupMap, String propertyPath) {
|
||||
GroupingContext(Map<ExpressionId, GroupByKey> groupMap) {
|
||||
this.groupMap = groupMap;
|
||||
this.groupPath = propertyPath;
|
||||
|
||||
aggNames = new ArrayList<>(groupMap.size());
|
||||
for (GroupingAgg gAgg : groupMap.values()) {
|
||||
aggNames.add(gAgg.id());
|
||||
GroupByKey lastAgg = null;
|
||||
for (Entry<ExpressionId, GroupByKey> entry : groupMap.entrySet()) {
|
||||
lastAgg = entry.getValue();
|
||||
}
|
||||
|
||||
Iterator<Entry<ExpressionId, GroupingAgg>> iterator = groupMap.entrySet().iterator();
|
||||
|
||||
Entry<ExpressionId, GroupingAgg> entry = iterator.next();
|
||||
headAggId = entry.getKey();
|
||||
head = entry.getValue();
|
||||
|
||||
GroupingAgg lastAgg = head;
|
||||
|
||||
while (iterator.hasNext()) {
|
||||
lastAgg = iterator.next().getValue();
|
||||
}
|
||||
tail = lastAgg;
|
||||
}
|
||||
|
||||
|
||||
GroupingAgg groupFor(Expression exp) {
|
||||
GroupByKey groupFor(Expression exp) {
|
||||
if (Functions.isAggregate(exp)) {
|
||||
AggregateFunction f = (AggregateFunction) exp;
|
||||
// if there's at least one agg in the tree
|
||||
if (groupPath != null) {
|
||||
GroupingAgg matchingGroup = null;
|
||||
if (!groupMap.isEmpty()) {
|
||||
GroupByKey matchingGroup = null;
|
||||
// group found - finding the dedicated agg
|
||||
if (f.field() instanceof NamedExpression) {
|
||||
matchingGroup = groupMap.get(((NamedExpression) f.field()).id());
|
||||
|
@ -239,18 +218,16 @@ abstract class QueryTranslator {
|
|||
}
|
||||
}
|
||||
|
||||
// creates a tree of GroupBy aggs plus some extra information
|
||||
// useful for tree validation/group referencing
|
||||
/**
|
||||
* Creates the list of GroupBy keys
|
||||
*/
|
||||
static GroupingContext groupBy(List<? extends Expression> groupings) {
|
||||
if (groupings.isEmpty()) {
|
||||
return null;
|
||||
}
|
||||
|
||||
Map<ExpressionId, GroupingAgg> aggMap = new LinkedHashMap<>();
|
||||
Map<ExpressionId, GroupByKey> aggMap = new LinkedHashMap<>();
|
||||
|
||||
// nested the aggs but also
|
||||
// identify each agg by an expression for later referencing
|
||||
String propertyPath = "";
|
||||
for (Expression exp : groupings) {
|
||||
String aggId;
|
||||
if (exp instanceof NamedExpression) {
|
||||
|
@ -265,38 +242,36 @@ abstract class QueryTranslator {
|
|||
}
|
||||
aggId = ne.id().toString();
|
||||
|
||||
propertyPath = AggPath.path(propertyPath, aggId);
|
||||
|
||||
GroupingAgg agg = null;
|
||||
GroupByKey key = null;
|
||||
|
||||
// handle functions differently
|
||||
if (exp instanceof Function) {
|
||||
// dates are handled differently because of date histograms
|
||||
if (exp instanceof DateTimeHistogramFunction) {
|
||||
DateTimeHistogramFunction dthf = (DateTimeHistogramFunction) exp;
|
||||
agg = new GroupByDateAgg(aggId, AggPath.bucketValue(propertyPath), nameOf(exp),
|
||||
dthf.interval(), dthf.timeZone());
|
||||
key = new GroupByDateKey(aggId, nameOf(exp), dthf.interval(), dthf.timeZone());
|
||||
}
|
||||
// all other scalar functions become a script
|
||||
else if (exp instanceof ScalarFunction) {
|
||||
ScalarFunction sf = (ScalarFunction) exp;
|
||||
agg = new GroupByScriptAgg(aggId, AggPath.bucketValue(propertyPath), nameOf(exp), sf.asScript());
|
||||
key = new GroupByScriptKey(aggId, nameOf(exp), sf.asScript());
|
||||
}
|
||||
// bumped into into an invalid function (which should be caught by the verifier)
|
||||
else {
|
||||
throw new SqlIllegalArgumentException("Cannot GROUP BY function {}", exp);
|
||||
}
|
||||
}
|
||||
else {
|
||||
agg = new GroupByColumnAgg(aggId, AggPath.bucketValue(propertyPath), ne.name());
|
||||
key = new GroupByColumnKey(aggId, ne.name());
|
||||
}
|
||||
|
||||
aggMap.put(ne.id(), agg);
|
||||
aggMap.put(ne.id(), key);
|
||||
}
|
||||
else {
|
||||
throw new SqlIllegalArgumentException("Don't know how to group on {}", exp.nodeString());
|
||||
}
|
||||
}
|
||||
return new GroupingContext(aggMap, propertyPath);
|
||||
return new GroupingContext(aggMap);
|
||||
}
|
||||
|
||||
static QueryTranslation and(Location loc, QueryTranslation left, QueryTranslation right) {
|
||||
|
@ -630,7 +605,7 @@ abstract class QueryTranslator {
|
|||
return new TermQuery(loc, name, value);
|
||||
}
|
||||
|
||||
Check.isTrue(false, "don't know how to translate binary comparison %s in %s", bc.right().nodeString(), bc);
|
||||
Check.isTrue(false, "don't know how to translate binary comparison [{}] in [{}]", bc.right().nodeString(), bc);
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
@ -739,91 +714,91 @@ abstract class QueryTranslator {
|
|||
static class DistinctCounts extends SingleValueAggTranslator<Count> {
|
||||
|
||||
@Override
|
||||
protected LeafAgg toAgg(String id, String path, Count c) {
|
||||
protected LeafAgg toAgg(String id, Count c) {
|
||||
if (!c.distinct()) {
|
||||
return null;
|
||||
}
|
||||
return new CardinalityAgg(id, path, field(c));
|
||||
return new CardinalityAgg(id, field(c));
|
||||
}
|
||||
}
|
||||
|
||||
static class Sums extends SingleValueAggTranslator<Sum> {
|
||||
|
||||
@Override
|
||||
protected LeafAgg toAgg(String id, String path, Sum s) {
|
||||
return new SumAgg(id, path, field(s));
|
||||
protected LeafAgg toAgg(String id, Sum s) {
|
||||
return new SumAgg(id, field(s));
|
||||
}
|
||||
}
|
||||
|
||||
static class Avgs extends SingleValueAggTranslator<Avg> {
|
||||
|
||||
@Override
|
||||
protected LeafAgg toAgg(String id, String path, Avg a) {
|
||||
return new AvgAgg(id, path, field(a));
|
||||
protected LeafAgg toAgg(String id, Avg a) {
|
||||
return new AvgAgg(id, field(a));
|
||||
}
|
||||
}
|
||||
|
||||
static class Maxes extends SingleValueAggTranslator<Max> {
|
||||
|
||||
@Override
|
||||
protected LeafAgg toAgg(String id, String path, Max m) {
|
||||
return new MaxAgg(id, path, field(m));
|
||||
protected LeafAgg toAgg(String id, Max m) {
|
||||
return new MaxAgg(id, field(m));
|
||||
}
|
||||
}
|
||||
|
||||
static class Mins extends SingleValueAggTranslator<Min> {
|
||||
|
||||
@Override
|
||||
protected LeafAgg toAgg(String id, String path, Min m) {
|
||||
return new MinAgg(id, path, field(m));
|
||||
protected LeafAgg toAgg(String id, Min m) {
|
||||
return new MinAgg(id, field(m));
|
||||
}
|
||||
}
|
||||
|
||||
static class StatsAggs extends CompoundAggTranslator<Stats> {
|
||||
|
||||
@Override
|
||||
protected LeafAgg toAgg(String id, String path, Stats s) {
|
||||
return new StatsAgg(id, path, field(s));
|
||||
protected LeafAgg toAgg(String id, Stats s) {
|
||||
return new StatsAgg(id, field(s));
|
||||
}
|
||||
}
|
||||
|
||||
static class ExtendedStatsAggs extends CompoundAggTranslator<ExtendedStats> {
|
||||
|
||||
@Override
|
||||
protected LeafAgg toAgg(String id, String path, ExtendedStats e) {
|
||||
return new ExtendedStatsAgg(id, path, field(e));
|
||||
protected LeafAgg toAgg(String id, ExtendedStats e) {
|
||||
return new ExtendedStatsAgg(id, field(e));
|
||||
}
|
||||
}
|
||||
|
||||
static class MatrixStatsAggs extends CompoundAggTranslator<MatrixStats> {
|
||||
|
||||
@Override
|
||||
protected LeafAgg toAgg(String id, String path, MatrixStats m) {
|
||||
return new MatrixStatsAgg(id, path, singletonList(field(m)));
|
||||
protected LeafAgg toAgg(String id, MatrixStats m) {
|
||||
return new MatrixStatsAgg(id, singletonList(field(m)));
|
||||
}
|
||||
}
|
||||
|
||||
static class PercentilesAggs extends CompoundAggTranslator<Percentiles> {
|
||||
|
||||
@Override
|
||||
protected LeafAgg toAgg(String id, String path, Percentiles p) {
|
||||
return new PercentilesAgg(id, path, field(p), doubleValuesOf(p.percents()));
|
||||
protected LeafAgg toAgg(String id, Percentiles p) {
|
||||
return new PercentilesAgg(id, field(p), doubleValuesOf(p.percents()));
|
||||
}
|
||||
}
|
||||
|
||||
static class PercentileRanksAggs extends CompoundAggTranslator<PercentileRanks> {
|
||||
|
||||
@Override
|
||||
protected LeafAgg toAgg(String id, String path, PercentileRanks p) {
|
||||
return new PercentileRanksAgg(id, path, field(p), doubleValuesOf(p.values()));
|
||||
protected LeafAgg toAgg(String id, PercentileRanks p) {
|
||||
return new PercentileRanksAgg(id, field(p), doubleValuesOf(p.values()));
|
||||
}
|
||||
}
|
||||
|
||||
static class DateTimes extends SingleValueAggTranslator<Min> {
|
||||
|
||||
@Override
|
||||
protected LeafAgg toAgg(String id, String path, Min m) {
|
||||
return new MinAgg(id, path, field(m));
|
||||
protected LeafAgg toAgg(String id, Min m) {
|
||||
return new MinAgg(id, field(m));
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -832,33 +807,31 @@ abstract class QueryTranslator {
|
|||
private final Class<?> typeToken = ReflectionUtils.detectSuperTypeForRuleLike(getClass());
|
||||
|
||||
@SuppressWarnings("unchecked")
|
||||
public final LeafAgg apply(String id, String parent, Function f) {
|
||||
return (typeToken.isInstance(f) ? asAgg(id, parent, (F) f) : null);
|
||||
public final LeafAgg apply(String id, Function f) {
|
||||
return (typeToken.isInstance(f) ? asAgg(id, (F) f) : null);
|
||||
}
|
||||
|
||||
protected abstract LeafAgg asAgg(String id, String parent, F f);
|
||||
protected abstract LeafAgg asAgg(String id, F f);
|
||||
}
|
||||
|
||||
abstract static class SingleValueAggTranslator<F extends Function> extends AggTranslator<F> {
|
||||
|
||||
@Override
|
||||
protected final LeafAgg asAgg(String id, String parent, F function) {
|
||||
String path = parent == null ? id : AggPath.path(parent, id);
|
||||
return toAgg(id, AggPath.metricValue(path), function);
|
||||
protected final LeafAgg asAgg(String id, F function) {
|
||||
return toAgg(id, function);
|
||||
}
|
||||
|
||||
protected abstract LeafAgg toAgg(String id, String path, F f);
|
||||
protected abstract LeafAgg toAgg(String id, F f);
|
||||
}
|
||||
|
||||
abstract static class CompoundAggTranslator<C extends CompoundNumericAggregate> extends AggTranslator<C> {
|
||||
|
||||
@Override
|
||||
protected final LeafAgg asAgg(String id, String parent, C function) {
|
||||
String path = parent == null ? id : AggPath.path(parent, id);
|
||||
return toAgg(id, path, function);
|
||||
protected final LeafAgg asAgg(String id, C function) {
|
||||
return toAgg(id, function);
|
||||
}
|
||||
|
||||
protected abstract LeafAgg toAgg(String id, String path, C f);
|
||||
protected abstract LeafAgg toAgg(String id, C f);
|
||||
}
|
||||
|
||||
|
||||
|
|
|
@ -7,6 +7,7 @@ package org.elasticsearch.xpack.sql.plugin;
|
|||
|
||||
import org.elasticsearch.action.ActionListener;
|
||||
import org.elasticsearch.client.Client;
|
||||
import org.elasticsearch.common.io.stream.NamedWriteableRegistry;
|
||||
import org.elasticsearch.common.io.stream.StreamInput;
|
||||
import org.elasticsearch.common.io.stream.StreamOutput;
|
||||
import org.elasticsearch.xpack.sql.session.Configuration;
|
||||
|
@ -57,8 +58,8 @@ public class CliFormatterCursor implements Cursor {
|
|||
}
|
||||
|
||||
@Override
|
||||
public void nextPage(Configuration cfg, Client client, ActionListener<RowSet> listener) {
|
||||
delegate.nextPage(cfg, client, listener);
|
||||
public void nextPage(Configuration cfg, Client client, NamedWriteableRegistry registry, ActionListener<RowSet> listener) {
|
||||
delegate.nextPage(cfg, client, registry, listener);
|
||||
}
|
||||
|
||||
@Override
|
||||
|
@ -73,8 +74,12 @@ public class CliFormatterCursor implements Cursor {
|
|||
|
||||
@Override
|
||||
public boolean equals(Object o) {
|
||||
if (this == o) return true;
|
||||
if (o == null || getClass() != o.getClass()) return false;
|
||||
if (this == o) {
|
||||
return true;
|
||||
}
|
||||
if (o == null || getClass() != o.getClass()) {
|
||||
return false;
|
||||
}
|
||||
CliFormatterCursor that = (CliFormatterCursor) o;
|
||||
return Objects.equals(delegate, that.delegate) &&
|
||||
Objects.equals(formatter, that.formatter);
|
||||
|
|
|
@ -20,6 +20,7 @@ import org.elasticsearch.rest.RestStatus;
|
|||
import org.elasticsearch.rest.action.RestResponseListener;
|
||||
import org.elasticsearch.rest.action.RestToXContentListener;
|
||||
import org.elasticsearch.xpack.sql.session.Cursor;
|
||||
import org.elasticsearch.xpack.sql.session.Cursors;
|
||||
|
||||
import java.io.IOException;
|
||||
import java.nio.charset.StandardCharsets;
|
||||
|
@ -63,7 +64,7 @@ public class RestSqlQueryAction extends BaseRestHandler {
|
|||
return channel -> client.execute(SqlQueryAction.INSTANCE, sqlRequest, new RestResponseListener<SqlQueryResponse>(channel) {
|
||||
@Override
|
||||
public RestResponse buildResponse(SqlQueryResponse response) throws Exception {
|
||||
Cursor cursor = Cursor.decodeFromString(sqlRequest.cursor());
|
||||
Cursor cursor = Cursors.decodeFromString(sqlRequest.cursor());
|
||||
final String data = text.format(cursor, request, response);
|
||||
|
||||
RestResponse restResponse = new BytesRestResponse(RestStatus.OK, text.contentType(request),
|
||||
|
@ -72,7 +73,7 @@ public class RestSqlQueryAction extends BaseRestHandler {
|
|||
Cursor responseCursor = text.wrapCursor(cursor, response);
|
||||
|
||||
if (responseCursor != Cursor.EMPTY) {
|
||||
restResponse.addHeader("Cursor", Cursor.encodeToString(Version.CURRENT, responseCursor));
|
||||
restResponse.addHeader("Cursor", Cursors.encodeToString(Version.CURRENT, responseCursor));
|
||||
}
|
||||
restResponse.addHeader("Took-nanos", Long.toString(System.nanoTime() - startNanos));
|
||||
|
||||
|
|
|
@ -45,7 +45,6 @@ public class SqlPlugin extends Plugin implements ActionPlugin {
|
|||
|
||||
private final boolean enabled;
|
||||
private final SqlLicenseChecker sqlLicenseChecker;
|
||||
private IndexResolver indexResolver;
|
||||
|
||||
SqlPlugin(boolean enabled, SqlLicenseChecker sqlLicenseChecker) {
|
||||
this.enabled = enabled;
|
||||
|
@ -79,18 +78,19 @@ public class SqlPlugin extends Plugin implements ActionPlugin {
|
|||
ResourceWatcherService resourceWatcherService, ScriptService scriptService,
|
||||
NamedXContentRegistry xContentRegistry, Environment environment,
|
||||
NodeEnvironment nodeEnvironment, NamedWriteableRegistry namedWriteableRegistry) {
|
||||
return createComponents(client, clusterService.getClusterName().value());
|
||||
|
||||
return createComponents(client, clusterService.getClusterName().value(), namedWriteableRegistry);
|
||||
}
|
||||
|
||||
/**
|
||||
* Create components used by the sql plugin.
|
||||
*/
|
||||
Collection<Object> createComponents(Client client, String clusterName) {
|
||||
Collection<Object> createComponents(Client client, String clusterName, NamedWriteableRegistry namedWriteableRegistry) {
|
||||
if (false == enabled) {
|
||||
return emptyList();
|
||||
}
|
||||
indexResolver = new IndexResolver(client, clusterName);
|
||||
return Arrays.asList(sqlLicenseChecker, indexResolver, new PlanExecutor(client, indexResolver));
|
||||
IndexResolver indexResolver = new IndexResolver(client, clusterName);
|
||||
return Arrays.asList(sqlLicenseChecker, indexResolver, new PlanExecutor(client, indexResolver, namedWriteableRegistry));
|
||||
}
|
||||
|
||||
@Override
|
||||
|
|
|
@ -8,6 +8,7 @@ package org.elasticsearch.xpack.sql.plugin;
|
|||
import org.elasticsearch.common.Strings;
|
||||
import org.elasticsearch.rest.RestRequest;
|
||||
import org.elasticsearch.xpack.sql.session.Cursor;
|
||||
import org.elasticsearch.xpack.sql.session.Cursors;
|
||||
import org.elasticsearch.xpack.sql.util.StringUtils;
|
||||
|
||||
import java.util.List;
|
||||
|
@ -232,7 +233,7 @@ enum TextFormat {
|
|||
}
|
||||
|
||||
Cursor wrapCursor(Cursor oldCursor, SqlQueryResponse response) {
|
||||
return Cursor.decodeFromString(response.cursor());
|
||||
return Cursors.decodeFromString(response.cursor());
|
||||
}
|
||||
|
||||
static TextFormat fromMediaTypeOrFormat(RestRequest request) {
|
||||
|
|
|
@ -16,6 +16,7 @@ import org.elasticsearch.transport.TransportService;
|
|||
import org.elasticsearch.xpack.sql.execution.PlanExecutor;
|
||||
import org.elasticsearch.xpack.sql.session.Configuration;
|
||||
import org.elasticsearch.xpack.sql.session.Cursor;
|
||||
import org.elasticsearch.xpack.sql.session.Cursors;
|
||||
|
||||
import static org.elasticsearch.xpack.sql.plugin.SqlClearCursorAction.NAME;
|
||||
|
||||
|
@ -38,7 +39,12 @@ public class TransportSqlClearCursorAction extends HandledTransportAction<SqlCle
|
|||
@Override
|
||||
protected void doExecute(SqlClearCursorRequest request, ActionListener<SqlClearCursorResponse> listener) {
|
||||
sqlLicenseChecker.checkIfSqlAllowed(request.mode());
|
||||
Cursor cursor = Cursor.decodeFromString(request.getCursor());
|
||||
operation(planExecutor, request, listener);
|
||||
}
|
||||
|
||||
public static void operation(PlanExecutor planExecutor, SqlClearCursorRequest request,
|
||||
ActionListener<SqlClearCursorResponse> listener) {
|
||||
Cursor cursor = Cursors.decodeFromString(request.getCursor());
|
||||
planExecutor.cleanCursor(Configuration.DEFAULT, cursor, ActionListener.wrap(
|
||||
success -> listener.onResponse(new SqlClearCursorResponse(success)), listener::onFailure));
|
||||
}
|
||||
|
|
|
@ -17,7 +17,7 @@ import org.elasticsearch.threadpool.ThreadPool;
|
|||
import org.elasticsearch.transport.TransportService;
|
||||
import org.elasticsearch.xpack.sql.execution.PlanExecutor;
|
||||
import org.elasticsearch.xpack.sql.session.Configuration;
|
||||
import org.elasticsearch.xpack.sql.session.Cursor;
|
||||
import org.elasticsearch.xpack.sql.session.Cursors;
|
||||
import org.elasticsearch.xpack.sql.session.RowSet;
|
||||
import org.elasticsearch.xpack.sql.session.SchemaRowSet;
|
||||
import org.elasticsearch.xpack.sql.type.Schema;
|
||||
|
@ -64,7 +64,7 @@ public class TransportSqlQueryAction extends HandledTransportAction<SqlQueryRequ
|
|||
planExecutor.sql(cfg, request.query(), request.params(),
|
||||
ActionListener.wrap(rowSet -> listener.onResponse(createResponse(request, rowSet)), listener::onFailure));
|
||||
} else {
|
||||
planExecutor.nextPage(cfg, Cursor.decodeFromString(request.cursor()),
|
||||
planExecutor.nextPage(cfg, Cursors.decodeFromString(request.cursor()),
|
||||
ActionListener.wrap(rowSet -> listener.onResponse(createResponse(rowSet, null)), listener::onFailure));
|
||||
}
|
||||
}
|
||||
|
@ -92,7 +92,7 @@ public class TransportSqlQueryAction extends HandledTransportAction<SqlQueryRequ
|
|||
});
|
||||
|
||||
return new SqlQueryResponse(
|
||||
Cursor.encodeToString(Version.CURRENT, rowSet.nextPageCursor()),
|
||||
Cursors.encodeToString(Version.CURRENT, rowSet.nextPageCursor()),
|
||||
columns,
|
||||
rows);
|
||||
}
|
||||
|
|
|
@ -8,22 +8,18 @@ package org.elasticsearch.xpack.sql.querydsl.agg;
|
|||
import java.util.Locale;
|
||||
import java.util.Objects;
|
||||
|
||||
import org.elasticsearch.search.aggregations.AggregationBuilder;
|
||||
import org.elasticsearch.xpack.sql.util.StringUtils;
|
||||
|
||||
import static java.lang.String.format;
|
||||
|
||||
/**
|
||||
* Base class holding common properties for Elasticsearch aggregations.
|
||||
*/
|
||||
public abstract class Agg {
|
||||
|
||||
private final String id;
|
||||
private final String fieldName;
|
||||
private final String propertyPath;
|
||||
private final String asParentPath;
|
||||
|
||||
Agg(String id, String propertyPath, String fieldName) {
|
||||
Agg(String id, String fieldName) {
|
||||
this.id = id;
|
||||
this.propertyPath = propertyPath;
|
||||
int index = propertyPath.lastIndexOf(".");
|
||||
this.asParentPath = index > 0 ? propertyPath.substring(0, index) : StringUtils.EMPTY;
|
||||
this.fieldName = fieldName;
|
||||
}
|
||||
|
||||
|
@ -31,23 +27,13 @@ public abstract class Agg {
|
|||
return id;
|
||||
}
|
||||
|
||||
public String propertyPath() {
|
||||
return propertyPath;
|
||||
}
|
||||
|
||||
public String asParentPath() {
|
||||
return asParentPath;
|
||||
}
|
||||
|
||||
public String fieldName() {
|
||||
return fieldName;
|
||||
}
|
||||
|
||||
abstract AggregationBuilder toBuilder();
|
||||
|
||||
@Override
|
||||
public int hashCode() {
|
||||
return Objects.hash(id, propertyPath);
|
||||
return Objects.hash(id, fieldName);
|
||||
}
|
||||
|
||||
@Override
|
||||
|
@ -62,12 +48,11 @@ public abstract class Agg {
|
|||
|
||||
Agg other = (Agg) obj;
|
||||
return Objects.equals(id, other.id)
|
||||
&& Objects.equals(fieldName, other.fieldName)
|
||||
&& Objects.equals(propertyPath, other.propertyPath);
|
||||
&& Objects.equals(fieldName, other.fieldName);
|
||||
}
|
||||
|
||||
@Override
|
||||
public String toString() {
|
||||
return format(Locale.ROOT, "%s(%s)#%s", getClass().getSimpleName(), fieldName, propertyPath);
|
||||
return format(Locale.ROOT, "%s(%s)", getClass().getSimpleName(), fieldName);
|
||||
}
|
||||
}
|
|
@ -1,102 +0,0 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License;
|
||||
* you may not use this file except in compliance with the Elastic License.
|
||||
*/
|
||||
package org.elasticsearch.xpack.sql.querydsl.agg;
|
||||
|
||||
import org.elasticsearch.common.Strings;
|
||||
import org.elasticsearch.xpack.sql.util.StringUtils;
|
||||
|
||||
// utility class around constructing and parsing ES specific aggregation path
|
||||
// see org.elasticsearch.search.aggregations.support.AggregationPath
|
||||
public abstract class AggPath {
|
||||
|
||||
private static final char PATH_DELIMITER_CHAR = '>';
|
||||
private static final String PATH_DELIMITER = String.valueOf(PATH_DELIMITER_CHAR);
|
||||
private static final String VALUE_DELIMITER = ".";
|
||||
private static final String PATH_BUCKET_VALUE = "._key";
|
||||
private static final String PATH_BUCKET_COUNT = "._count";
|
||||
private static final String PATH_BUCKET_VALUE_FORMATTED = "._key_as_string";
|
||||
private static final String PATH_DEFAULT_VALUE = ".value";
|
||||
|
||||
public static String bucketCount(String aggPath) {
|
||||
return aggPath + PATH_BUCKET_COUNT;
|
||||
}
|
||||
|
||||
public static String bucketValue(String aggPath) {
|
||||
return aggPath + PATH_BUCKET_VALUE;
|
||||
}
|
||||
|
||||
public static boolean isBucketValueFormatted(String path) {
|
||||
return path.endsWith(PATH_BUCKET_VALUE_FORMATTED);
|
||||
}
|
||||
|
||||
public static String bucketValueWithoutFormat(String path) {
|
||||
return path.substring(0, path.length() - PATH_BUCKET_VALUE_FORMATTED.length());
|
||||
}
|
||||
|
||||
public static String metricValue(String aggPath) {
|
||||
return aggPath + PATH_DEFAULT_VALUE;
|
||||
}
|
||||
|
||||
public static String metricValue(String aggPath, String valueName) {
|
||||
// handle aggPath inconsistency (for percentiles and percentileRanks) percentile[99.9] (valid) vs percentile.99.9 (invalid)
|
||||
return valueName.startsWith("[") ? aggPath + valueName : aggPath + VALUE_DELIMITER + valueName;
|
||||
}
|
||||
|
||||
public static String path(String parent, String child) {
|
||||
return (Strings.hasLength(parent) ? parent + PATH_DELIMITER : StringUtils.EMPTY) + child;
|
||||
}
|
||||
|
||||
/*
|
||||
* The depth indicates the level of an agg excluding the root agg
|
||||
* (because all aggs in SQL require a group). However all other
|
||||
* bucket aggs are counted.
|
||||
*
|
||||
* Since the path does not indicate the type of agg used, to
|
||||
* differentiate between metric properties and bucket properties,
|
||||
* the bucket value is considered. This is needed since one might
|
||||
* refer to the keys or count of a bucket path. As the opposite
|
||||
* side there are metric aggs which have the same level as their
|
||||
* parent (their nesting is an ES implementation detail).
|
||||
*
|
||||
* Examples:
|
||||
*
|
||||
* agg1 = 0; agg1 = default/root group
|
||||
* agg1>agg2._count = 1; ._count indicates agg2 is a bucket
|
||||
* agg and thus it counted - agg1 (default group),
|
||||
* depth=0, agg2 (bucketed), depth=1
|
||||
* agg1>agg2>agg3.value = 1; agg3.value indicates a metric bucket
|
||||
* thus only agg1 and agg2 are counted -> depth=2. In
|
||||
* other words, agg3.value has the same depth as
|
||||
* agg2._count
|
||||
* agg1>agg2>agg3._count = 2; ._count indicates agg3 is a
|
||||
* bucket agg, so count it for depth -> depth = 2
|
||||
* agg1>agg2>agg3.sum = 1; .sum indicates agg3 is a metric agg,
|
||||
* only agg1 and agg2 are bucket and with agg1 being the
|
||||
* default group -> depth = 1
|
||||
*/
|
||||
public static int depth(String path) {
|
||||
int depth = countCharIn(path, PATH_DELIMITER_CHAR);
|
||||
// a metric value always has .foo while a bucket prop with ._foo
|
||||
int dot = path.lastIndexOf(".");
|
||||
if (depth > 0 && dot > 0) {
|
||||
String prop = path.substring(dot + 1);
|
||||
if (!prop.startsWith("_")) {
|
||||
return Math.max(0, depth - 1);
|
||||
}
|
||||
}
|
||||
return depth;
|
||||
}
|
||||
|
||||
private static int countCharIn(CharSequence sequence, char c) {
|
||||
int count = 0;
|
||||
for (int i = 0; i < sequence.length(); i++) {
|
||||
if (c == sequence.charAt(i)) {
|
||||
count++;
|
||||
}
|
||||
}
|
||||
return count;
|
||||
}
|
||||
}
|
|
@ -6,8 +6,11 @@
|
|||
package org.elasticsearch.xpack.sql.querydsl.agg;
|
||||
|
||||
import org.elasticsearch.search.aggregations.AggregationBuilder;
|
||||
import org.elasticsearch.search.aggregations.PipelineAggregationBuilder;
|
||||
import org.elasticsearch.search.aggregations.bucket.composite.CompositeAggregationBuilder;
|
||||
import org.elasticsearch.search.aggregations.bucket.composite.CompositeValuesSourceBuilder;
|
||||
import org.elasticsearch.search.aggregations.bucket.filter.FiltersAggregationBuilder;
|
||||
import org.elasticsearch.xpack.sql.SqlIllegalArgumentException;
|
||||
import org.elasticsearch.xpack.sql.querydsl.container.Sort.Direction;
|
||||
|
||||
import java.util.ArrayList;
|
||||
import java.util.Collection;
|
||||
|
@ -15,185 +18,128 @@ import java.util.List;
|
|||
import java.util.Objects;
|
||||
|
||||
import static java.util.Collections.emptyList;
|
||||
import static org.elasticsearch.index.query.QueryBuilders.matchAllQuery;
|
||||
import static org.elasticsearch.xpack.sql.util.CollectionUtils.combine;
|
||||
import static org.elasticsearch.xpack.sql.util.StringUtils.EMPTY;
|
||||
|
||||
// Aggregations associated with a query.
|
||||
//
|
||||
// This maps the SQL target namely:
|
||||
// - aggregations are either at the root (default or nameless group) or belong to a named group
|
||||
// - groups are chained in a linear fashion (group1 -> group2 -> group3)
|
||||
//
|
||||
// Aggregations are of two types - concrete metric aggs or pipeline/filtering aggs
|
||||
//
|
||||
// Pipeline Notes:
|
||||
//
|
||||
// As pipeline aggs can refers to aggs across different groups (different tree levels)
|
||||
// one needs to pay attention attention before adding them to properly find the lowest level and
|
||||
// make everything relative to that.
|
||||
//
|
||||
// For example sum > 100 && count < 10 might imply sum referring on level 2 while count is on level 3
|
||||
// In this case, the pipeline would be added to group/level 2 (and refer to sum) while count will be
|
||||
// made relative to this group/level 3 namely group3>count
|
||||
|
||||
/**
|
||||
* SQL Aggregations associated with a query.
|
||||
*
|
||||
* This class maps the SQL GroupBy's (and co) to ES composite agg.
|
||||
* While the composite agg doesn't require a dedicated structure, for folding purposes, this structure
|
||||
* tracks the relationship between each key and its sub-aggs or pipelines.
|
||||
*
|
||||
* Since sub-aggs can only refer to their group key and these are on the root-level, the tree can have at most
|
||||
* 2 levels - the grouping and its sub-aggs.
|
||||
*
|
||||
* In case no group is specified (which maps to the default group in SQL), due to ES nature a 'dummy' filter agg
|
||||
* is used.
|
||||
*/
|
||||
public class Aggs {
|
||||
|
||||
private final List<GroupingAgg> groups;
|
||||
private final List<Agg> rootAggs;
|
||||
private final List<PipelineAgg> rootPipelineAggs;
|
||||
public static final String ROOT_GROUP_NAME = "groupby";
|
||||
|
||||
public static final GroupByKey IMPLICIT_GROUP_KEY = new GroupByKey(ROOT_GROUP_NAME, EMPTY, null) {
|
||||
|
||||
@Override
|
||||
public CompositeValuesSourceBuilder<?> asValueSource() {
|
||||
throw new SqlIllegalArgumentException("Default group does not translate to an aggregation");
|
||||
}
|
||||
|
||||
@Override
|
||||
protected GroupByKey copy(String id, String fieldName, Direction direction) {
|
||||
return this;
|
||||
}
|
||||
};
|
||||
|
||||
private final List<GroupByKey> groups;
|
||||
private final List<LeafAgg> metricAggs;
|
||||
private final List<PipelineAgg> pipelineAggs;
|
||||
|
||||
public Aggs() {
|
||||
this(emptyList(), emptyList(), emptyList());
|
||||
}
|
||||
|
||||
public Aggs(List<Agg> rootAggs, List<PipelineAgg> rootPipelineAggs, List<GroupingAgg> groups) {
|
||||
this.rootAggs = rootAggs;
|
||||
this.rootPipelineAggs = rootPipelineAggs;
|
||||
public Aggs(List<GroupByKey> groups, List<LeafAgg> metricAggs, List<PipelineAgg> pipelineAggs) {
|
||||
this.groups = groups;
|
||||
|
||||
this.metricAggs = metricAggs;
|
||||
this.pipelineAggs = pipelineAggs;
|
||||
}
|
||||
|
||||
public List<GroupingAgg> groups() {
|
||||
public List<GroupByKey> groups() {
|
||||
return groups;
|
||||
}
|
||||
|
||||
public List<AggregationBuilder> asAggBuilders() {
|
||||
List<AggregationBuilder> aggBuilders = new ArrayList<>(rootAggs.size() + 1);
|
||||
public AggregationBuilder asAggBuilder() {
|
||||
AggregationBuilder rootGroup = null;
|
||||
|
||||
for (Agg agg : rootAggs) {
|
||||
aggBuilders.add(agg.toBuilder());
|
||||
if (groups.isEmpty() && metricAggs.isEmpty()) {
|
||||
return null;
|
||||
}
|
||||
|
||||
// if there's a group, move everything under the composite agg
|
||||
if (!groups.isEmpty()) {
|
||||
AggregationBuilder root = null, last = null;
|
||||
for (GroupingAgg groupBy : groups) {
|
||||
AggregationBuilder builder = groupBy.toBuilder();
|
||||
if (root == null) {
|
||||
root = builder;
|
||||
}
|
||||
// chain each group
|
||||
if (last != null) {
|
||||
last.subAggregation(builder);
|
||||
}
|
||||
last = builder;
|
||||
List<CompositeValuesSourceBuilder<?>> keys = new ArrayList<>(groups.size());
|
||||
// first iterate to compute the sources
|
||||
for (GroupByKey key : groups) {
|
||||
keys.add(key.asValueSource());
|
||||
}
|
||||
|
||||
aggBuilders.add(root);
|
||||
rootGroup = new CompositeAggregationBuilder(ROOT_GROUP_NAME, keys);
|
||||
|
||||
} else {
|
||||
rootGroup = new FiltersAggregationBuilder(ROOT_GROUP_NAME, matchAllQuery());
|
||||
}
|
||||
|
||||
return aggBuilders;
|
||||
for (LeafAgg agg : metricAggs) {
|
||||
rootGroup.subAggregation(agg.toBuilder());
|
||||
}
|
||||
|
||||
public List<PipelineAggregationBuilder> asPipelineBuilders() {
|
||||
List<PipelineAggregationBuilder> aggBuilders = new ArrayList<>(rootPipelineAggs.size() + 1);
|
||||
|
||||
for (PipelineAgg agg : rootPipelineAggs) {
|
||||
aggBuilders.add(agg.toBuilder());
|
||||
for (PipelineAgg agg : pipelineAggs) {
|
||||
rootGroup.subAggregation(agg.toBuilder());
|
||||
}
|
||||
|
||||
return aggBuilders;
|
||||
return rootGroup;
|
||||
}
|
||||
|
||||
public Aggs addGroups(Collection<GroupingAgg> groups) {
|
||||
return new Aggs(rootAggs, rootPipelineAggs, combine(this.groups, groups));
|
||||
public boolean useImplicitGroupBy() {
|
||||
return groups.isEmpty();
|
||||
}
|
||||
|
||||
public Aggs addAgg(Agg agg) {
|
||||
if (agg instanceof GroupingAgg) {
|
||||
GroupingAgg group = (GroupingAgg) agg;
|
||||
return with(combine(this.groups, group));
|
||||
}
|
||||
if (agg instanceof LeafAgg) {
|
||||
LeafAgg leaf = (LeafAgg) agg;
|
||||
if (groups.isEmpty()) {
|
||||
return new Aggs(combine(rootAggs, leaf), rootPipelineAggs, this.groups);
|
||||
}
|
||||
else {
|
||||
// copy the groups and replace the tail
|
||||
List<GroupingAgg> groups = new ArrayList<>(this.groups);
|
||||
GroupingAgg tail = groups.get(groups.size() - 1);
|
||||
groups.set(groups.size() - 1, tail.withAggs(combine(tail.subAggs(), leaf)));
|
||||
return with(groups);
|
||||
}
|
||||
public Aggs addGroups(Collection<GroupByKey> groups) {
|
||||
return new Aggs(combine(this.groups, groups), metricAggs, pipelineAggs);
|
||||
}
|
||||
|
||||
throw new SqlIllegalArgumentException("Does not know how to handle type {}", agg);
|
||||
public Aggs addAgg(LeafAgg agg) {
|
||||
return new Aggs(groups, combine(metricAggs, agg), pipelineAggs);
|
||||
}
|
||||
|
||||
public Aggs addAgg(PipelineAgg pipelineAgg) {
|
||||
if (groups.isEmpty()) {
|
||||
return new Aggs(rootAggs, combine(rootPipelineAggs, pipelineAgg), this.groups);
|
||||
}
|
||||
else {
|
||||
// copy the groups and replace the tail
|
||||
List<GroupingAgg> groups = new ArrayList<>(this.groups);
|
||||
GroupingAgg tail = groups.get(groups.size() - 1);
|
||||
groups.set(groups.size() - 1, tail.withPipelines(combine(tail.subPipelines(), pipelineAgg)));
|
||||
return with(groups);
|
||||
}
|
||||
return new Aggs(groups, metricAggs, combine(pipelineAggs, pipelineAgg));
|
||||
}
|
||||
|
||||
|
||||
public Aggs addAgg(String groupId, LeafAgg child) {
|
||||
// it's a root agg
|
||||
if (groupId == null) {
|
||||
return new Aggs(combine(rootAggs, child), rootPipelineAggs, groups());
|
||||
}
|
||||
|
||||
List<GroupingAgg> groups = new ArrayList<>(this.groups);
|
||||
|
||||
for (int i = 0; i < groups.size(); i++) {
|
||||
GroupingAgg group = groups.get(i);
|
||||
if (groupId.equals(group.id())) {
|
||||
groups.set(i, group.withAggs(combine(group.subAggs(), child)));
|
||||
return with(groups);
|
||||
}
|
||||
}
|
||||
throw new SqlIllegalArgumentException("Could not find group named {}", groupId);
|
||||
}
|
||||
|
||||
public Aggs addAgg(String groupId, PipelineAgg child) {
|
||||
// it's a root agg
|
||||
if (groupId == null) {
|
||||
return new Aggs(rootAggs, combine(rootPipelineAggs, child), groups());
|
||||
}
|
||||
|
||||
List<GroupingAgg> groups = new ArrayList<>(this.groups);
|
||||
|
||||
for (int i = 0; i < groups.size(); i++) {
|
||||
GroupingAgg group = groups.get(i);
|
||||
if (groupId.equals(group.id())) {
|
||||
groups.set(i, group.withPipelines(combine(group.subPipelines(), child)));
|
||||
return with(groups);
|
||||
}
|
||||
}
|
||||
throw new SqlIllegalArgumentException("Could not find group named {}", groupId);
|
||||
}
|
||||
|
||||
public GroupingAgg findGroupForAgg(String groupOrAggId) {
|
||||
for (GroupingAgg group : this.groups) {
|
||||
public GroupByKey findGroupForAgg(String groupOrAggId) {
|
||||
for (GroupByKey group : this.groups) {
|
||||
if (groupOrAggId.equals(group.id())) {
|
||||
return group;
|
||||
}
|
||||
for (LeafAgg subAgg : group.subAggs()) {
|
||||
if (groupOrAggId.equals(subAgg.id())) {
|
||||
return group;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// is it maybe a root agg
|
||||
for (Agg agg : rootAggs) {
|
||||
// maybe it's the default group agg ?
|
||||
for (Agg agg : metricAggs) {
|
||||
if (groupOrAggId.equals(agg.id())) {
|
||||
return GroupingAgg.DEFAULT_GROUP;
|
||||
return IMPLICIT_GROUP_KEY;
|
||||
}
|
||||
}
|
||||
|
||||
return null;
|
||||
}
|
||||
|
||||
public Aggs updateGroup(GroupingAgg group) {
|
||||
List<GroupingAgg> groups = new ArrayList<>(this.groups);
|
||||
public Aggs updateGroup(GroupByKey group) {
|
||||
List<GroupByKey> groups = new ArrayList<>(this.groups);
|
||||
for (int i = 0; i < groups.size(); i++) {
|
||||
GroupingAgg g = groups.get(i);
|
||||
GroupByKey g = groups.get(i);
|
||||
if (group.id().equals(g.id())) {
|
||||
groups.set(i, group);
|
||||
return with(groups);
|
||||
|
@ -202,13 +148,13 @@ public class Aggs {
|
|||
throw new SqlIllegalArgumentException("Could not find group named {}", group.id());
|
||||
}
|
||||
|
||||
public Aggs with(List<GroupingAgg> groups) {
|
||||
return new Aggs(rootAggs, rootPipelineAggs, groups);
|
||||
public Aggs with(List<GroupByKey> groups) {
|
||||
return new Aggs(groups, metricAggs, pipelineAggs);
|
||||
}
|
||||
|
||||
@Override
|
||||
public int hashCode() {
|
||||
return Objects.hash(rootAggs, rootPipelineAggs, groups);
|
||||
return Objects.hash(groups, metricAggs, pipelineAggs);
|
||||
}
|
||||
|
||||
@Override
|
||||
|
@ -222,8 +168,9 @@ public class Aggs {
|
|||
}
|
||||
|
||||
Aggs other = (Aggs) obj;
|
||||
return Objects.equals(rootAggs, other.rootAggs)
|
||||
&& Objects.equals(rootPipelineAggs, other.rootPipelineAggs)
|
||||
&& Objects.equals(groups, other.groups);
|
||||
return Objects.equals(groups, other.groups)
|
||||
&& Objects.equals(metricAggs, other.metricAggs)
|
||||
&& Objects.equals(pipelineAggs, other.pipelineAggs);
|
||||
|
||||
}
|
||||
}
|
|
@ -11,8 +11,8 @@ import static org.elasticsearch.search.aggregations.AggregationBuilders.avg;
|
|||
|
||||
public class AvgAgg extends LeafAgg {
|
||||
|
||||
public AvgAgg(String id, String propertyPath, String fieldName) {
|
||||
super(id, propertyPath, fieldName);
|
||||
public AvgAgg(String id, String fieldName) {
|
||||
super(id, fieldName);
|
||||
}
|
||||
|
||||
@Override
|
||||
|
|
|
@ -11,8 +11,8 @@ import static org.elasticsearch.search.aggregations.AggregationBuilders.cardinal
|
|||
|
||||
public class CardinalityAgg extends LeafAgg {
|
||||
|
||||
public CardinalityAgg(String id, String propertyPath, String fieldName) {
|
||||
super(id, propertyPath, fieldName);
|
||||
public CardinalityAgg(String id, String fieldName) {
|
||||
super(id, fieldName);
|
||||
}
|
||||
|
||||
@Override AggregationBuilder toBuilder() {
|
||||
|
|
|
@ -11,8 +11,8 @@ import static org.elasticsearch.search.aggregations.AggregationBuilders.extended
|
|||
|
||||
public class ExtendedStatsAgg extends LeafAgg {
|
||||
|
||||
public ExtendedStatsAgg(String id, String propertyPath, String fieldName) {
|
||||
super(id, propertyPath, fieldName);
|
||||
public ExtendedStatsAgg(String id, String fieldName) {
|
||||
super(id, fieldName);
|
||||
}
|
||||
|
||||
@Override
|
||||
|
|
|
@ -1,86 +0,0 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License;
|
||||
* you may not use this file except in compliance with the Elastic License.
|
||||
*/
|
||||
package org.elasticsearch.xpack.sql.querydsl.agg;
|
||||
|
||||
import org.elasticsearch.search.aggregations.AggregationBuilder;
|
||||
import org.elasticsearch.search.aggregations.BucketOrder;
|
||||
import org.elasticsearch.search.aggregations.bucket.terms.TermsAggregationBuilder;
|
||||
import org.elasticsearch.xpack.sql.querydsl.container.Sort;
|
||||
import org.elasticsearch.xpack.sql.querydsl.container.Sort.Direction;
|
||||
|
||||
import java.util.ArrayList;
|
||||
import java.util.List;
|
||||
import java.util.Map;
|
||||
import java.util.Map.Entry;
|
||||
|
||||
import static java.util.Collections.emptyList;
|
||||
import static java.util.Collections.emptyMap;
|
||||
import static org.elasticsearch.search.aggregations.AggregationBuilders.terms;
|
||||
|
||||
public class GroupByColumnAgg extends GroupingAgg {
|
||||
|
||||
private static final int DEFAULT_LIMIT = 512;
|
||||
private final int limit;
|
||||
|
||||
public GroupByColumnAgg(String id, String propertyPath, String fieldName) {
|
||||
this(id, propertyPath, fieldName, emptyList(), emptyList(), emptyMap(), -1);
|
||||
}
|
||||
|
||||
public GroupByColumnAgg(String id, String propertyPath, String fieldName, List<LeafAgg> subAggs,
|
||||
List<PipelineAgg> subPipelines, Map<String, Direction> order, int limit) {
|
||||
super(id, propertyPath, fieldName, subAggs, subPipelines, order);
|
||||
this.limit = limit < 0 ? DEFAULT_LIMIT : Math.min(limit, DEFAULT_LIMIT);
|
||||
}
|
||||
|
||||
public int limit() {
|
||||
return limit;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected AggregationBuilder toGroupingAgg() {
|
||||
// TODO: the size should be configurable
|
||||
TermsAggregationBuilder terms = termsTarget(terms(id()).size(limit));
|
||||
|
||||
List<BucketOrder> termOrders = emptyList();
|
||||
if (!order().isEmpty()) {
|
||||
termOrders = new ArrayList<>();
|
||||
for (Entry<String, Sort.Direction> entry : order().entrySet()) {
|
||||
String key = entry.getKey();
|
||||
boolean asc = entry.getValue() == Direction.ASC;
|
||||
BucketOrder o = null;
|
||||
// special cases
|
||||
if (GROUP_KEY_SORTING.equals(key)) {
|
||||
o = BucketOrder.key(asc);
|
||||
}
|
||||
else if (GROUP_COUNT_SORTING.equals(key)) {
|
||||
o = BucketOrder.count(asc);
|
||||
}
|
||||
else {
|
||||
o = BucketOrder.aggregation(key, asc);
|
||||
}
|
||||
termOrders.add(o);
|
||||
}
|
||||
terms.order(termOrders);
|
||||
}
|
||||
|
||||
terms.minDocCount(1);
|
||||
return terms;
|
||||
}
|
||||
|
||||
protected TermsAggregationBuilder termsTarget(TermsAggregationBuilder builder) {
|
||||
return builder.field(fieldName());
|
||||
}
|
||||
|
||||
@Override
|
||||
protected GroupByColumnAgg copy(String id, String propertyPath, String fieldName, List<LeafAgg> subAggs,
|
||||
List<PipelineAgg> subPipelines, Map<String, Direction> order) {
|
||||
return new GroupByColumnAgg(id, propertyPath, fieldName, subAggs, subPipelines, order, limit);
|
||||
}
|
||||
|
||||
public GroupByColumnAgg withLimit(int limit) {
|
||||
return new GroupByColumnAgg(id(), propertyPath(), fieldName(), subAggs(), subPipelines(), order(), limit);
|
||||
}
|
||||
}
|
|
@ -0,0 +1,35 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License;
|
||||
* you may not use this file except in compliance with the Elastic License.
|
||||
*/
|
||||
package org.elasticsearch.xpack.sql.querydsl.agg;
|
||||
|
||||
import org.elasticsearch.search.aggregations.bucket.composite.TermsValuesSourceBuilder;
|
||||
import org.elasticsearch.xpack.sql.querydsl.container.Sort.Direction;
|
||||
|
||||
/**
|
||||
* GROUP BY key for regular fields.
|
||||
*/
|
||||
public class GroupByColumnKey extends GroupByKey {
|
||||
|
||||
public GroupByColumnKey(String id, String fieldName) {
|
||||
this(id, fieldName, null);
|
||||
}
|
||||
|
||||
public GroupByColumnKey(String id, String fieldName, Direction direction) {
|
||||
super(id, fieldName, direction);
|
||||
}
|
||||
|
||||
@Override
|
||||
public TermsValuesSourceBuilder asValueSource() {
|
||||
return new TermsValuesSourceBuilder(id())
|
||||
.field(fieldName())
|
||||
.order(direction().asOrder());
|
||||
}
|
||||
|
||||
@Override
|
||||
protected GroupByKey copy(String id, String fieldName, Direction direction) {
|
||||
return new GroupByColumnKey(id, fieldName, direction);
|
||||
}
|
||||
}
|
|
@ -1,77 +0,0 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License;
|
||||
* you may not use this file except in compliance with the Elastic License.
|
||||
*/
|
||||
package org.elasticsearch.xpack.sql.querydsl.agg;
|
||||
|
||||
import org.elasticsearch.search.aggregations.AggregationBuilder;
|
||||
import org.elasticsearch.search.aggregations.BucketOrder;
|
||||
import org.elasticsearch.search.aggregations.bucket.histogram.DateHistogramAggregationBuilder;
|
||||
import org.elasticsearch.search.aggregations.bucket.histogram.DateHistogramInterval;
|
||||
import org.elasticsearch.xpack.sql.querydsl.container.Sort;
|
||||
import org.elasticsearch.xpack.sql.querydsl.container.Sort.Direction;
|
||||
import org.joda.time.DateTimeZone;
|
||||
|
||||
import java.util.List;
|
||||
import java.util.Map;
|
||||
import java.util.Map.Entry;
|
||||
|
||||
import static java.util.Collections.emptyList;
|
||||
import static java.util.Collections.emptyMap;
|
||||
import static org.elasticsearch.search.aggregations.AggregationBuilders.dateHistogram;
|
||||
|
||||
public class GroupByDateAgg extends GroupingAgg {
|
||||
|
||||
private final String interval;
|
||||
private final DateTimeZone timeZone;
|
||||
|
||||
public GroupByDateAgg(String id, String propertyPath, String fieldName, String interval, DateTimeZone timeZone) {
|
||||
this(id, propertyPath, fieldName, interval, timeZone, emptyList(), emptyList(), emptyMap());
|
||||
}
|
||||
|
||||
public GroupByDateAgg(String id, String propertyPath, String fieldName, String interval, DateTimeZone timeZone,
|
||||
List<LeafAgg> subAggs, List<PipelineAgg> subPipelines, Map<String, Direction> order) {
|
||||
super(id, propertyPath, fieldName, subAggs, subPipelines, order);
|
||||
this.interval = interval;
|
||||
this.timeZone = timeZone;
|
||||
}
|
||||
|
||||
public String interval() {
|
||||
return interval;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected AggregationBuilder toGroupingAgg() {
|
||||
DateHistogramAggregationBuilder dhab = dateHistogram(id())
|
||||
.field(fieldName())
|
||||
.timeZone(timeZone)
|
||||
.dateHistogramInterval(new DateHistogramInterval(interval));
|
||||
if (!order().isEmpty()) {
|
||||
for (Entry<String, Sort.Direction> entry : order().entrySet()) {
|
||||
String key = entry.getKey();
|
||||
boolean asc = entry.getValue() == Direction.ASC;
|
||||
// special cases
|
||||
if (GROUP_KEY_SORTING.equals(key)) {
|
||||
dhab.order(BucketOrder.key(asc));
|
||||
}
|
||||
else if (GROUP_COUNT_SORTING.equals(key)) {
|
||||
dhab.order(BucketOrder.count(asc));
|
||||
}
|
||||
else {
|
||||
dhab.order(BucketOrder.aggregation(key, asc));
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
dhab.minDocCount(1);
|
||||
|
||||
return dhab;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected GroupingAgg copy(String id, String propertyPath, String fieldName, List<LeafAgg> subAggs,
|
||||
List<PipelineAgg> subPipelines, Map<String, Direction> order) {
|
||||
return new GroupByDateAgg(id, propertyPath, fieldName, interval, timeZone, subAggs, subPipelines, order);
|
||||
}
|
||||
}
|
|
@ -0,0 +1,68 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License;
|
||||
* you may not use this file except in compliance with the Elastic License.
|
||||
*/
|
||||
package org.elasticsearch.xpack.sql.querydsl.agg;
|
||||
|
||||
import org.elasticsearch.search.aggregations.bucket.composite.DateHistogramValuesSourceBuilder;
|
||||
import org.elasticsearch.search.aggregations.bucket.histogram.DateHistogramInterval;
|
||||
import org.elasticsearch.xpack.sql.querydsl.container.Sort.Direction;
|
||||
import org.joda.time.DateTimeZone;
|
||||
|
||||
import java.util.Objects;
|
||||
|
||||
/**
|
||||
* GROUP BY key specific for date fields.
|
||||
*/
|
||||
public class GroupByDateKey extends GroupByKey {
|
||||
|
||||
private final String interval;
|
||||
private final DateTimeZone timeZone;
|
||||
|
||||
public GroupByDateKey(String id, String fieldName, String interval, DateTimeZone timeZone) {
|
||||
this(id, fieldName, null, interval, timeZone);
|
||||
}
|
||||
|
||||
public GroupByDateKey(String id, String fieldName, Direction direction, String interval, DateTimeZone timeZone) {
|
||||
super(id, fieldName, direction);
|
||||
this.interval = interval;
|
||||
this.timeZone = timeZone;
|
||||
}
|
||||
|
||||
public String interval() {
|
||||
return interval;
|
||||
}
|
||||
|
||||
public DateTimeZone timeZone() {
|
||||
return timeZone;
|
||||
}
|
||||
|
||||
@Override
|
||||
public DateHistogramValuesSourceBuilder asValueSource() {
|
||||
return new DateHistogramValuesSourceBuilder(id())
|
||||
.field(fieldName())
|
||||
.dateHistogramInterval(new DateHistogramInterval(interval))
|
||||
.timeZone(timeZone);
|
||||
}
|
||||
|
||||
@Override
|
||||
protected GroupByKey copy(String id, String fieldName, Direction direction) {
|
||||
return new GroupByDateKey(id, fieldName, direction, interval, timeZone);
|
||||
}
|
||||
|
||||
@Override
|
||||
public int hashCode() {
|
||||
return Objects.hash(super.hashCode(), interval, timeZone);
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean equals(Object obj) {
|
||||
if (super.equals(obj)) {
|
||||
GroupByDateKey other = (GroupByDateKey) obj;
|
||||
return Objects.equals(interval, other.interval)
|
||||
&& Objects.equals(timeZone, other.timeZone);
|
||||
}
|
||||
return false;
|
||||
}
|
||||
}
|
|
@ -0,0 +1,47 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License;
|
||||
* you may not use this file except in compliance with the Elastic License.
|
||||
*/
|
||||
package org.elasticsearch.xpack.sql.querydsl.agg;
|
||||
|
||||
import org.elasticsearch.search.aggregations.bucket.composite.CompositeValuesSourceBuilder;
|
||||
import org.elasticsearch.xpack.sql.querydsl.container.Sort.Direction;
|
||||
|
||||
import java.util.Objects;
|
||||
|
||||
/**
|
||||
* A key for a SQL GroupBy which maps to value source for composite aggregation.
|
||||
*/
|
||||
public abstract class GroupByKey extends Agg {
|
||||
|
||||
private final Direction direction;
|
||||
|
||||
GroupByKey(String id, String fieldName, Direction direction) {
|
||||
super(id, fieldName);
|
||||
// ASC is the default order of CompositeValueSource
|
||||
this.direction = direction == null ? Direction.ASC : direction;
|
||||
}
|
||||
|
||||
public Direction direction() {
|
||||
return direction;
|
||||
}
|
||||
|
||||
public abstract CompositeValuesSourceBuilder<?> asValueSource();
|
||||
|
||||
protected abstract GroupByKey copy(String id, String fieldName, Direction direction);
|
||||
|
||||
public GroupByKey with(Direction direction) {
|
||||
return this.direction == direction ? this : copy(id(), fieldName(), direction);
|
||||
}
|
||||
|
||||
@Override
|
||||
public int hashCode() {
|
||||
return Objects.hash(id(), fieldName(), direction);
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean equals(Object obj) {
|
||||
return super.equals(obj) && Objects.equals(direction, ((GroupByKey) obj).direction);
|
||||
}
|
||||
}
|
|
@ -1,57 +0,0 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License;
|
||||
* you may not use this file except in compliance with the Elastic License.
|
||||
*/
|
||||
package org.elasticsearch.xpack.sql.querydsl.agg;
|
||||
|
||||
import org.elasticsearch.search.aggregations.bucket.terms.TermsAggregationBuilder;
|
||||
import org.elasticsearch.search.aggregations.support.ValueType;
|
||||
import org.elasticsearch.xpack.sql.expression.function.scalar.script.ScriptTemplate;
|
||||
import org.elasticsearch.xpack.sql.querydsl.container.Sort.Direction;
|
||||
|
||||
import java.util.List;
|
||||
import java.util.Map;
|
||||
|
||||
import static java.util.Collections.emptyList;
|
||||
import static java.util.Collections.emptyMap;
|
||||
|
||||
public class GroupByScriptAgg extends GroupByColumnAgg {
|
||||
|
||||
private final ScriptTemplate script;
|
||||
|
||||
public GroupByScriptAgg(String id, String propertyPath, String fieldName, ScriptTemplate script) {
|
||||
this(id, propertyPath, fieldName, script, emptyList(), emptyList(), emptyMap(), -1);
|
||||
}
|
||||
|
||||
public GroupByScriptAgg(String id, String propertyPath, String fieldName, ScriptTemplate script, List<LeafAgg> subAggs,
|
||||
List<PipelineAgg> subPipelines, Map<String, Direction> order, int limit) {
|
||||
super(id, propertyPath, fieldName, subAggs, subPipelines, order, limit);
|
||||
this.script = script;
|
||||
}
|
||||
|
||||
public ScriptTemplate script() {
|
||||
return script;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected TermsAggregationBuilder termsTarget(TermsAggregationBuilder builder) {
|
||||
builder.script(script.toPainless());
|
||||
if (script.outputType().isNumeric()) {
|
||||
builder.valueType(ValueType.NUMBER);
|
||||
}
|
||||
|
||||
return builder;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected GroupByScriptAgg copy(String id, String propertyPath, String fieldName, List<LeafAgg> subAggs,
|
||||
List<PipelineAgg> subPipelines, Map<String, Direction> order) {
|
||||
return new GroupByScriptAgg(id, propertyPath, fieldName, script, subAggs, subPipelines, order, limit());
|
||||
}
|
||||
|
||||
@Override
|
||||
public GroupByScriptAgg withLimit(int limit) {
|
||||
return new GroupByScriptAgg(id(), propertyPath(), fieldName(), script, subAggs(), subPipelines(), order(), limit);
|
||||
}
|
||||
}
|
|
@ -0,0 +1,62 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License;
|
||||
* you may not use this file except in compliance with the Elastic License.
|
||||
*/
|
||||
package org.elasticsearch.xpack.sql.querydsl.agg;
|
||||
|
||||
import org.elasticsearch.search.aggregations.bucket.composite.TermsValuesSourceBuilder;
|
||||
import org.elasticsearch.search.aggregations.support.ValueType;
|
||||
import org.elasticsearch.xpack.sql.expression.function.scalar.script.ScriptTemplate;
|
||||
import org.elasticsearch.xpack.sql.querydsl.container.Sort.Direction;
|
||||
|
||||
import java.util.Objects;
|
||||
|
||||
/**
|
||||
* GROUP BY key for scripts (typically caused by functions).
|
||||
*/
|
||||
public class GroupByScriptKey extends GroupByKey {
|
||||
|
||||
private final ScriptTemplate script;
|
||||
|
||||
public GroupByScriptKey(String id, String fieldName, ScriptTemplate script) {
|
||||
this(id, fieldName, null, script);
|
||||
}
|
||||
|
||||
public GroupByScriptKey(String id, String fieldName, Direction direction, ScriptTemplate script) {
|
||||
super(id, fieldName, direction);
|
||||
this.script = script;
|
||||
}
|
||||
|
||||
public ScriptTemplate script() {
|
||||
return script;
|
||||
}
|
||||
|
||||
@Override
|
||||
public TermsValuesSourceBuilder asValueSource() {
|
||||
TermsValuesSourceBuilder builder = new TermsValuesSourceBuilder(id())
|
||||
.script(script.toPainless())
|
||||
.order(direction().asOrder());
|
||||
|
||||
if (script.outputType().isNumeric()) {
|
||||
builder.valueType(ValueType.NUMBER);
|
||||
}
|
||||
|
||||
return builder;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected GroupByKey copy(String id, String fieldName, Direction direction) {
|
||||
return new GroupByScriptKey(id, fieldName, direction, script);
|
||||
}
|
||||
|
||||
@Override
|
||||
public int hashCode() {
|
||||
return Objects.hash(super.hashCode(), script);
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean equals(Object obj) {
|
||||
return super.equals(obj) && Objects.equals(script, ((GroupByScriptKey) obj).script);
|
||||
}
|
||||
}
|
|
@ -1,133 +0,0 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License;
|
||||
* you may not use this file except in compliance with the Elastic License.
|
||||
*/
|
||||
package org.elasticsearch.xpack.sql.querydsl.agg;
|
||||
|
||||
import org.elasticsearch.search.aggregations.AggregationBuilder;
|
||||
import org.elasticsearch.xpack.sql.SqlIllegalArgumentException;
|
||||
import org.elasticsearch.xpack.sql.querydsl.container.Sort.Direction;
|
||||
|
||||
import java.util.LinkedHashMap;
|
||||
import java.util.List;
|
||||
import java.util.Map;
|
||||
import java.util.Objects;
|
||||
|
||||
import static java.util.Collections.emptyList;
|
||||
import static org.elasticsearch.xpack.sql.util.StringUtils.EMPTY;
|
||||
|
||||
public abstract class GroupingAgg extends Agg {
|
||||
|
||||
protected static final String GROUP_KEY_SORTING = "_key";
|
||||
protected static final String GROUP_COUNT_SORTING = "_count";
|
||||
|
||||
public static final GroupingAgg DEFAULT_GROUP = new GroupingAgg(EMPTY, EMPTY, EMPTY, emptyList(), emptyList(), null) {
|
||||
|
||||
@Override
|
||||
protected AggregationBuilder toGroupingAgg() {
|
||||
throw new SqlIllegalArgumentException("Default group cannot build aggregations");
|
||||
}
|
||||
|
||||
@Override
|
||||
protected GroupingAgg copy(String id, String propertyPath, String fieldName, List<LeafAgg> subAggs,
|
||||
List<PipelineAgg> subPipelines, Map<String, Direction> order) {
|
||||
throw new SqlIllegalArgumentException("Default group cannot be cloned");
|
||||
}
|
||||
};
|
||||
|
||||
private final List<LeafAgg> subAggs;
|
||||
private final List<PipelineAgg> subPipelines;
|
||||
private final Map<String, Direction> order;
|
||||
|
||||
GroupingAgg(String id, String propertyPath, String fieldName, List<LeafAgg> subAggs,
|
||||
List<PipelineAgg> subPipelines, Map<String, Direction> order) {
|
||||
super(id, propertyPath, fieldName);
|
||||
this.subAggs = subAggs;
|
||||
this.subPipelines = subPipelines;
|
||||
this.order = order;
|
||||
}
|
||||
|
||||
public List<LeafAgg> subAggs() {
|
||||
return subAggs;
|
||||
}
|
||||
|
||||
public List<PipelineAgg> subPipelines() {
|
||||
return subPipelines;
|
||||
}
|
||||
|
||||
public Map<String, Direction> order() {
|
||||
return order;
|
||||
}
|
||||
|
||||
@Override
|
||||
AggregationBuilder toBuilder() {
|
||||
AggregationBuilder groupingAgg = toGroupingAgg();
|
||||
for (LeafAgg leafAgg : subAggs) {
|
||||
groupingAgg.subAggregation(leafAgg.toBuilder());
|
||||
}
|
||||
for (PipelineAgg aggFilter : subPipelines) {
|
||||
groupingAgg.subAggregation(aggFilter.toBuilder());
|
||||
}
|
||||
return groupingAgg;
|
||||
}
|
||||
|
||||
protected abstract AggregationBuilder toGroupingAgg();
|
||||
|
||||
|
||||
public GroupingAgg withAggs(List<LeafAgg> subAggs) {
|
||||
return copy(id(), propertyPath(), fieldName(), subAggs, subPipelines, order);
|
||||
}
|
||||
|
||||
public GroupingAgg withPipelines(List<PipelineAgg> subPipelines) {
|
||||
return copy(id(), propertyPath(), fieldName(), subAggs, subPipelines, order);
|
||||
}
|
||||
|
||||
public GroupingAgg with(String id) {
|
||||
return Objects.equals(id(), id) ? this : copy(id, propertyPath(), fieldName(), subAggs, subPipelines, order);
|
||||
}
|
||||
|
||||
public GroupingAgg with(Direction order) {
|
||||
return with(GROUP_KEY_SORTING, order);
|
||||
}
|
||||
|
||||
public GroupingAgg with(String leafAggId, Direction order) {
|
||||
if (Objects.equals(this.order.get(leafAggId), order)) {
|
||||
return this;
|
||||
}
|
||||
Map<String, Direction> newOrder = new LinkedHashMap<>(this.order);
|
||||
newOrder.put(leafAggId, order);
|
||||
return copy(id(), propertyPath(), fieldName(), subAggs, subPipelines, newOrder);
|
||||
}
|
||||
|
||||
protected abstract GroupingAgg copy(String id, String propertyPath, String fieldName, List<LeafAgg> subAggs,
|
||||
List<PipelineAgg> subPipelines, Map<String, Direction> order);
|
||||
|
||||
@Override
|
||||
public int hashCode() {
|
||||
return Objects.hash(order, id(), propertyPath(), fieldName(), subAggs(), subPipelines());
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean equals(Object obj) {
|
||||
if (this == obj) {
|
||||
return true;
|
||||
}
|
||||
|
||||
if (obj == null || getClass() != obj.getClass()) {
|
||||
return false;
|
||||
}
|
||||
|
||||
GroupByColumnAgg other = (GroupByColumnAgg) obj;
|
||||
return Objects.equals(id(), other.id())
|
||||
&& Objects.equals(propertyPath(), other.propertyPath())
|
||||
&& Objects.equals(fieldName(), other.fieldName())
|
||||
&& Objects.equals(subAggs(), other.subAggs())
|
||||
&& Objects.equals(subPipelines(), other.subPipelines());
|
||||
}
|
||||
|
||||
@Override
|
||||
public String toString() {
|
||||
return super.toString() + "=" + subAggs() + "|" + subPipelines();
|
||||
}
|
||||
}
|
|
@ -5,9 +5,13 @@
|
|||
*/
|
||||
package org.elasticsearch.xpack.sql.querydsl.agg;
|
||||
|
||||
import org.elasticsearch.search.aggregations.AggregationBuilder;
|
||||
|
||||
public abstract class LeafAgg extends Agg {
|
||||
|
||||
LeafAgg(String id, String propertyPath, String fieldName) {
|
||||
super(id, propertyPath, fieldName);
|
||||
LeafAgg(String id, String fieldName) {
|
||||
super(id, fieldName);
|
||||
}
|
||||
|
||||
abstract AggregationBuilder toBuilder();
|
||||
}
|
||||
|
|
|
@ -15,8 +15,8 @@ public class MatrixStatsAgg extends LeafAgg {
|
|||
|
||||
private final List<String> fields;
|
||||
|
||||
public MatrixStatsAgg(String id, String propertyPath, List<String> fields) {
|
||||
super(id, propertyPath, "<multi-field>");
|
||||
public MatrixStatsAgg(String id, List<String> fields) {
|
||||
super(id, "<multi-field>");
|
||||
this.fields = fields;
|
||||
}
|
||||
|
||||
|
|
|
@ -11,8 +11,8 @@ import static org.elasticsearch.search.aggregations.AggregationBuilders.max;
|
|||
|
||||
public class MaxAgg extends LeafAgg {
|
||||
|
||||
public MaxAgg(String id, String propertyPath, String fieldName) {
|
||||
super(id, propertyPath, fieldName);
|
||||
public MaxAgg(String id, String fieldName) {
|
||||
super(id, fieldName);
|
||||
}
|
||||
|
||||
@Override
|
||||
|
|
|
@ -11,8 +11,8 @@ import static org.elasticsearch.search.aggregations.AggregationBuilders.min;
|
|||
|
||||
public class MinAgg extends LeafAgg {
|
||||
|
||||
public MinAgg(String id, String propertyPath, String fieldName) {
|
||||
super(id, propertyPath, fieldName);
|
||||
public MinAgg(String id, String fieldName) {
|
||||
super(id, fieldName);
|
||||
}
|
||||
|
||||
@Override
|
||||
|
|
|
@ -15,8 +15,8 @@ public class PercentileRanksAgg extends LeafAgg {
|
|||
|
||||
private final List<Double> values;
|
||||
|
||||
public PercentileRanksAgg(String id, String propertyPath, String fieldName, List<Double> values) {
|
||||
super(id, propertyPath, fieldName);
|
||||
public PercentileRanksAgg(String id, String fieldName, List<Double> values) {
|
||||
super(id, fieldName);
|
||||
this.values = values;
|
||||
}
|
||||
|
||||
|
|
|
@ -15,8 +15,8 @@ public class PercentilesAgg extends LeafAgg {
|
|||
|
||||
private final List<Double> percents;
|
||||
|
||||
public PercentilesAgg(String id, String propertyPath, String fieldName, List<Double> percents) {
|
||||
super(id, propertyPath, fieldName);
|
||||
public PercentilesAgg(String id, String fieldName, List<Double> percents) {
|
||||
super(id, fieldName);
|
||||
this.percents = percents;
|
||||
}
|
||||
|
||||
|
|
|
@ -11,8 +11,8 @@ import static org.elasticsearch.search.aggregations.AggregationBuilders.stats;
|
|||
|
||||
public class StatsAgg extends LeafAgg {
|
||||
|
||||
public StatsAgg(String id, String propertyPath, String fieldName) {
|
||||
super(id, propertyPath, fieldName);
|
||||
public StatsAgg(String id, String fieldName) {
|
||||
super(id, fieldName);
|
||||
}
|
||||
|
||||
@Override
|
||||
|
|
|
@ -11,8 +11,8 @@ import static org.elasticsearch.search.aggregations.AggregationBuilders.sum;
|
|||
|
||||
public class SumAgg extends LeafAgg {
|
||||
|
||||
public SumAgg(String id, String propertyPath, String fieldName) {
|
||||
super(id, propertyPath, fieldName);
|
||||
public SumAgg(String id, String fieldName) {
|
||||
super(id, fieldName);
|
||||
}
|
||||
|
||||
@Override AggregationBuilder toBuilder() {
|
||||
|
|
|
@ -1,55 +0,0 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License;
|
||||
* you may not use this file except in compliance with the Elastic License.
|
||||
*/
|
||||
package org.elasticsearch.xpack.sql.querydsl.container;
|
||||
|
||||
import org.elasticsearch.xpack.sql.execution.search.FieldExtraction;
|
||||
import org.elasticsearch.xpack.sql.execution.search.SqlSourceBuilder;
|
||||
import org.elasticsearch.xpack.sql.querydsl.agg.AggPath;
|
||||
|
||||
public class AggRef implements FieldExtraction {
|
||||
private final String path;
|
||||
private final int depth;
|
||||
private final String innerKey;
|
||||
|
||||
public AggRef(String path) {
|
||||
this(path, null);
|
||||
}
|
||||
|
||||
public AggRef(String path, String innerKey) {
|
||||
this.path = path;
|
||||
depth = AggPath.depth(path);
|
||||
this.innerKey = innerKey;
|
||||
}
|
||||
|
||||
|
||||
@Override
|
||||
public String toString() {
|
||||
return path;
|
||||
}
|
||||
|
||||
@Override
|
||||
public int depth() {
|
||||
return depth;
|
||||
}
|
||||
|
||||
public String path() {
|
||||
return path;
|
||||
}
|
||||
|
||||
public String innerKey() {
|
||||
return innerKey;
|
||||
}
|
||||
|
||||
@Override
|
||||
public void collectFields(SqlSourceBuilder sourceBuilder) {
|
||||
// Aggregations do not need any special fields
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean supportedByAggsOnlyQuery() {
|
||||
return true;
|
||||
}
|
||||
}
|
|
@ -21,11 +21,6 @@ public class ComputedRef implements FieldExtraction {
|
|||
return processor;
|
||||
}
|
||||
|
||||
@Override
|
||||
public int depth() {
|
||||
return processor.depth();
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean supportedByAggsOnlyQuery() {
|
||||
return processor.supportedByAggsOnlyQuery();
|
||||
|
|
|
@ -15,11 +15,6 @@ public abstract class FieldReference implements FieldExtraction {
|
|||
*/
|
||||
public abstract String name();
|
||||
|
||||
@Override
|
||||
public final int depth() {
|
||||
return 0;
|
||||
}
|
||||
|
||||
@Override
|
||||
public final boolean supportedByAggsOnlyQuery() {
|
||||
return false;
|
||||
|
|
|
@ -0,0 +1,21 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License;
|
||||
* you may not use this file except in compliance with the Elastic License.
|
||||
*/
|
||||
package org.elasticsearch.xpack.sql.querydsl.container;
|
||||
|
||||
import org.elasticsearch.xpack.sql.execution.search.AggRef;
|
||||
|
||||
/**
|
||||
* Aggregation reference pointing to the (so called) global count, meaning
|
||||
* COUNT over the entire data set.
|
||||
*/
|
||||
public final class GlobalCountRef extends AggRef {
|
||||
public static final GlobalCountRef INSTANCE = new GlobalCountRef();
|
||||
|
||||
@Override
|
||||
public String toString() {
|
||||
return "#_Total_Hits_#";
|
||||
}
|
||||
}
|
|
@ -0,0 +1,50 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License;
|
||||
* you may not use this file except in compliance with the Elastic License.
|
||||
*/
|
||||
package org.elasticsearch.xpack.sql.querydsl.container;
|
||||
|
||||
import org.elasticsearch.xpack.sql.execution.search.AggRef;
|
||||
import org.joda.time.DateTimeZone;
|
||||
|
||||
/**
|
||||
* Reference to a GROUP BY agg (typically this gets translated to a composite key).
|
||||
*/
|
||||
public class GroupByRef extends AggRef {
|
||||
|
||||
public enum Property {
|
||||
VALUE, COUNT;
|
||||
}
|
||||
|
||||
private final String key;
|
||||
private final Property property;
|
||||
private final DateTimeZone timeZone;
|
||||
|
||||
public GroupByRef(String key) {
|
||||
this(key, null, null);
|
||||
}
|
||||
|
||||
public GroupByRef(String key, Property property, DateTimeZone timeZone) {
|
||||
this.key = key;
|
||||
this.property = property == null ? Property.VALUE : property;
|
||||
this.timeZone = timeZone;
|
||||
}
|
||||
|
||||
public String key() {
|
||||
return key;
|
||||
}
|
||||
|
||||
public Property property() {
|
||||
return property;
|
||||
}
|
||||
|
||||
public DateTimeZone timeZone() {
|
||||
return timeZone;
|
||||
}
|
||||
|
||||
@Override
|
||||
public String toString() {
|
||||
return "|" + key + (property == Property.COUNT ? ".count" : "") + "|";
|
||||
}
|
||||
}
|
|
@ -0,0 +1,52 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License;
|
||||
* you may not use this file except in compliance with the Elastic License.
|
||||
*/
|
||||
package org.elasticsearch.xpack.sql.querydsl.container;
|
||||
|
||||
import org.elasticsearch.xpack.sql.execution.search.AggRef;
|
||||
import org.elasticsearch.xpack.sql.querydsl.agg.Aggs;
|
||||
|
||||
/**
|
||||
* Reference to a sub/nested/metric aggregation.
|
||||
* Due to how ES query works, this is _always_ a child aggregation with the grouping (composite agg) as the parent.
|
||||
*/
|
||||
public class MetricAggRef extends AggRef {
|
||||
|
||||
private final String name;
|
||||
private final String property;
|
||||
private final String innerKey;
|
||||
|
||||
public MetricAggRef(String name) {
|
||||
this(name, "value");
|
||||
}
|
||||
|
||||
public MetricAggRef(String name, String property) {
|
||||
this(name, property, null);
|
||||
}
|
||||
|
||||
public MetricAggRef(String name, String property, String innerKey) {
|
||||
this.name = name;
|
||||
this.property = property;
|
||||
this.innerKey = innerKey;
|
||||
}
|
||||
|
||||
public String name() {
|
||||
return name;
|
||||
}
|
||||
|
||||
public String property() {
|
||||
return property;
|
||||
}
|
||||
|
||||
public String innerKey() {
|
||||
return innerKey;
|
||||
}
|
||||
|
||||
@Override
|
||||
public String toString() {
|
||||
String i = innerKey != null ? "[" + innerKey + "]" : "";
|
||||
return Aggs.ROOT_GROUP_NAME + ">" + name + "." + property + i;
|
||||
}
|
||||
}
|
|
@ -21,10 +21,10 @@ import org.elasticsearch.xpack.sql.expression.function.ScoreAttribute;
|
|||
import org.elasticsearch.xpack.sql.expression.function.scalar.ScalarFunctionAttribute;
|
||||
import org.elasticsearch.xpack.sql.expression.function.scalar.processor.definition.ProcessorDefinition;
|
||||
import org.elasticsearch.xpack.sql.expression.function.scalar.processor.definition.ScoreProcessorDefinition;
|
||||
import org.elasticsearch.xpack.sql.querydsl.agg.AggPath;
|
||||
import org.elasticsearch.xpack.sql.querydsl.agg.Aggs;
|
||||
import org.elasticsearch.xpack.sql.querydsl.agg.GroupingAgg;
|
||||
import org.elasticsearch.xpack.sql.querydsl.agg.GroupByKey;
|
||||
import org.elasticsearch.xpack.sql.querydsl.agg.LeafAgg;
|
||||
import org.elasticsearch.xpack.sql.querydsl.container.GroupByRef.Property;
|
||||
import org.elasticsearch.xpack.sql.querydsl.query.BoolQuery;
|
||||
import org.elasticsearch.xpack.sql.querydsl.query.MatchAll;
|
||||
import org.elasticsearch.xpack.sql.querydsl.query.NestedQuery;
|
||||
|
@ -60,7 +60,7 @@ public class QueryContainer {
|
|||
private final Map<Attribute, Attribute> aliases;
|
||||
|
||||
// pseudo functions (like count) - that are 'extracted' from other aggs
|
||||
private final Map<String, GroupingAgg> pseudoFunctions;
|
||||
private final Map<String, GroupByKey> pseudoFunctions;
|
||||
|
||||
// scalar function processors - recorded as functions get folded;
|
||||
// at scrolling, their inputs (leaves) get updated
|
||||
|
@ -71,14 +71,13 @@ public class QueryContainer {
|
|||
|
||||
// computed
|
||||
private final boolean aggsOnly;
|
||||
private final int aggDepth;
|
||||
|
||||
public QueryContainer() {
|
||||
this(null, null, null, null, null, null, null, -1);
|
||||
}
|
||||
|
||||
public QueryContainer(Query query, Aggs aggs, List<FieldExtraction> refs, Map<Attribute, Attribute> aliases,
|
||||
Map<String, GroupingAgg> pseudoFunctions,
|
||||
Map<String, GroupByKey> pseudoFunctions,
|
||||
Map<Attribute, ProcessorDefinition> scalarFunctions,
|
||||
Set<Sort> sort, int limit) {
|
||||
this.query = query;
|
||||
|
@ -90,7 +89,6 @@ public class QueryContainer {
|
|||
this.sort = sort == null || sort.isEmpty() ? emptySet() : sort;
|
||||
this.limit = limit;
|
||||
aggsOnly = columns.stream().allMatch(FieldExtraction::supportedByAggsOnlyQuery);
|
||||
aggDepth = columns.stream().mapToInt(FieldExtraction::depth).max().orElse(0);
|
||||
}
|
||||
|
||||
public Query query() {
|
||||
|
@ -109,7 +107,7 @@ public class QueryContainer {
|
|||
return aliases;
|
||||
}
|
||||
|
||||
public Map<String, GroupingAgg> pseudoFunctions() {
|
||||
public Map<String, GroupByKey> pseudoFunctions() {
|
||||
return pseudoFunctions;
|
||||
}
|
||||
|
||||
|
@ -125,10 +123,6 @@ public class QueryContainer {
|
|||
return aggsOnly;
|
||||
}
|
||||
|
||||
public int aggDepth() {
|
||||
return aggDepth;
|
||||
}
|
||||
|
||||
public boolean hasColumns() {
|
||||
return !columns.isEmpty();
|
||||
}
|
||||
|
@ -149,7 +143,7 @@ public class QueryContainer {
|
|||
return new QueryContainer(query, aggs, columns, a, pseudoFunctions, scalarFunctions, sort, limit);
|
||||
}
|
||||
|
||||
public QueryContainer withPseudoFunctions(Map<String, GroupingAgg> p) {
|
||||
public QueryContainer withPseudoFunctions(Map<String, GroupByKey> p) {
|
||||
return new QueryContainer(query, aggs, columns, aliases, p, scalarFunctions, sort, limit);
|
||||
}
|
||||
|
||||
|
@ -299,39 +293,28 @@ public class QueryContainer {
|
|||
//
|
||||
// agg methods
|
||||
//
|
||||
public QueryContainer addAggColumn(String aggPath) {
|
||||
return with(combine(columns, new AggRef(aggPath)));
|
||||
}
|
||||
|
||||
public QueryContainer addAggColumn(String aggPath, String innerKey) {
|
||||
return with(combine(columns, new AggRef(aggPath, innerKey)));
|
||||
}
|
||||
|
||||
public QueryContainer addAggCount(GroupingAgg parentGroup, String functionId) {
|
||||
FieldExtraction ref = parentGroup == null ? TotalCountRef.INSTANCE : new AggRef(AggPath.bucketCount(parentGroup.asParentPath()));
|
||||
Map<String, GroupingAgg> pseudoFunctions = new LinkedHashMap<>(this.pseudoFunctions);
|
||||
pseudoFunctions.put(functionId, parentGroup);
|
||||
public QueryContainer addAggCount(GroupByKey group, String functionId) {
|
||||
FieldExtraction ref = group == null ? GlobalCountRef.INSTANCE : new GroupByRef(group.id(), Property.COUNT, null);
|
||||
Map<String, GroupByKey> pseudoFunctions = new LinkedHashMap<>(this.pseudoFunctions);
|
||||
pseudoFunctions.put(functionId, group);
|
||||
return new QueryContainer(query, aggs, combine(columns, ref), aliases, pseudoFunctions, scalarFunctions, sort, limit);
|
||||
}
|
||||
|
||||
public QueryContainer addAgg(String groupId, LeafAgg agg) {
|
||||
return addAgg(groupId, agg, agg.propertyPath());
|
||||
return with(aggs.addAgg(agg));
|
||||
}
|
||||
|
||||
public QueryContainer addAgg(String groupId, LeafAgg agg, String aggRefPath) {
|
||||
return new QueryContainer(query, aggs.addAgg(groupId, agg), columns, aliases, pseudoFunctions, scalarFunctions, sort, limit);
|
||||
}
|
||||
|
||||
public QueryContainer addGroups(Collection<GroupingAgg> values) {
|
||||
public QueryContainer addGroups(Collection<GroupByKey> values) {
|
||||
return with(aggs.addGroups(values));
|
||||
}
|
||||
|
||||
public QueryContainer updateGroup(GroupingAgg group) {
|
||||
return with(aggs.updateGroup(group));
|
||||
public GroupByKey findGroupForAgg(String aggId) {
|
||||
return aggs.findGroupForAgg(aggId);
|
||||
}
|
||||
|
||||
public GroupingAgg findGroupForAgg(String aggId) {
|
||||
return aggs.findGroupForAgg(aggId);
|
||||
public QueryContainer updateGroup(GroupByKey group) {
|
||||
return with(aggs.updateGroup(group));
|
||||
}
|
||||
|
||||
//
|
||||
|
|
|
@ -5,6 +5,7 @@
|
|||
*/
|
||||
package org.elasticsearch.xpack.sql.querydsl.container;
|
||||
|
||||
import org.elasticsearch.search.sort.SortOrder;
|
||||
import org.elasticsearch.xpack.sql.expression.Order.OrderDirection;
|
||||
|
||||
public class Sort {
|
||||
|
@ -15,6 +16,10 @@ public class Sort {
|
|||
public static Direction from(OrderDirection dir) {
|
||||
return dir == null || dir == OrderDirection.ASC ? ASC : DESC;
|
||||
}
|
||||
|
||||
public SortOrder asOrder() {
|
||||
return this == Direction.ASC ? SortOrder.ASC : SortOrder.DESC;
|
||||
}
|
||||
}
|
||||
|
||||
private final Direction direction;
|
||||
|
|
|
@ -1,24 +0,0 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License;
|
||||
* you may not use this file except in compliance with the Elastic License.
|
||||
*/
|
||||
package org.elasticsearch.xpack.sql.querydsl.container;
|
||||
|
||||
import org.elasticsearch.xpack.sql.util.StringUtils;
|
||||
|
||||
// somewhat of a fake agg (since it gets optimized and it gets its value from the response)
|
||||
public final class TotalCountRef extends AggRef {
|
||||
public static final TotalCountRef INSTANCE = new TotalCountRef();
|
||||
|
||||
public static final String PATH = "#_count_#";
|
||||
|
||||
TotalCountRef() {
|
||||
super(StringUtils.EMPTY);
|
||||
}
|
||||
|
||||
@Override
|
||||
public String toString() {
|
||||
return "TotalCountRef";
|
||||
}
|
||||
}
|
|
@ -5,97 +5,24 @@
|
|||
*/
|
||||
package org.elasticsearch.xpack.sql.session;
|
||||
|
||||
import org.elasticsearch.Version;
|
||||
import org.elasticsearch.action.ActionListener;
|
||||
import org.elasticsearch.client.Client;
|
||||
import org.elasticsearch.common.io.stream.InputStreamStreamInput;
|
||||
import org.elasticsearch.common.io.stream.NamedWriteable;
|
||||
import org.elasticsearch.common.io.stream.NamedWriteableAwareStreamInput;
|
||||
import org.elasticsearch.common.io.stream.NamedWriteableRegistry;
|
||||
import org.elasticsearch.common.io.stream.OutputStreamStreamOutput;
|
||||
import org.elasticsearch.common.io.stream.StreamInput;
|
||||
import org.elasticsearch.common.io.stream.StreamOutput;
|
||||
import org.elasticsearch.xpack.sql.execution.search.ScrollCursor;
|
||||
import org.elasticsearch.xpack.sql.execution.search.extractor.HitExtractors;
|
||||
import org.elasticsearch.xpack.sql.plugin.CliFormatterCursor;
|
||||
|
||||
import java.io.ByteArrayInputStream;
|
||||
import java.io.ByteArrayOutputStream;
|
||||
import java.io.IOException;
|
||||
import java.io.OutputStream;
|
||||
import java.nio.charset.StandardCharsets;
|
||||
import java.util.ArrayList;
|
||||
import java.util.Base64;
|
||||
import java.util.List;
|
||||
|
||||
/**
|
||||
* Information required to access the next page of response.
|
||||
*/
|
||||
public interface Cursor extends NamedWriteable {
|
||||
NamedWriteableRegistry CURSOR_REGISTRY = new NamedWriteableRegistry(Cursor.getNamedWriteables());
|
||||
|
||||
Cursor EMPTY = EmptyCursor.INSTANCE;
|
||||
|
||||
/**
|
||||
* Request the next page of data.
|
||||
*/
|
||||
void nextPage(Configuration cfg, Client client, ActionListener<RowSet> listener);
|
||||
void nextPage(Configuration cfg, Client client, NamedWriteableRegistry registry, ActionListener<RowSet> listener);
|
||||
|
||||
/**
|
||||
* Cleans the resources associated with the cursor
|
||||
*/
|
||||
void clear(Configuration cfg, Client client, ActionListener<Boolean> listener);
|
||||
|
||||
/**
|
||||
* The {@link NamedWriteable}s required to deserialize {@link Cursor}s.
|
||||
*/
|
||||
static List<NamedWriteableRegistry.Entry> getNamedWriteables() {
|
||||
List<NamedWriteableRegistry.Entry> entries = new ArrayList<>();
|
||||
entries.addAll(HitExtractors.getNamedWriteables());
|
||||
entries.add(new NamedWriteableRegistry.Entry(Cursor.class, EmptyCursor.NAME, in -> EMPTY));
|
||||
entries.add(new NamedWriteableRegistry.Entry(Cursor.class, ScrollCursor.NAME, ScrollCursor::new));
|
||||
entries.add(new NamedWriteableRegistry.Entry(Cursor.class, CliFormatterCursor.NAME, CliFormatterCursor::new));
|
||||
return entries;
|
||||
}
|
||||
|
||||
/**
|
||||
* Write a {@linkplain Cursor} to a string for serialization across xcontent.
|
||||
*/
|
||||
static String encodeToString(Version version, Cursor info) {
|
||||
if(info == EMPTY) {
|
||||
return "";
|
||||
}
|
||||
try (ByteArrayOutputStream os = new ByteArrayOutputStream()) {
|
||||
try (OutputStream base64 = Base64.getEncoder().wrap(os);
|
||||
StreamOutput out = new OutputStreamStreamOutput(base64)) {
|
||||
Version.writeVersion(version, out);
|
||||
out.writeNamedWriteable(info);
|
||||
}
|
||||
return os.toString(StandardCharsets.UTF_8.name());
|
||||
} catch (IOException ex) {
|
||||
throw new RuntimeException("unexpected failure converting next page info to a string", ex);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Read a {@linkplain Cursor} from a string.
|
||||
*/
|
||||
static Cursor decodeFromString(String info) {
|
||||
if (info.isEmpty()) {
|
||||
return EMPTY;
|
||||
}
|
||||
byte[] bytes = info.getBytes(StandardCharsets.UTF_8);
|
||||
try (StreamInput delegate = new InputStreamStreamInput(Base64.getDecoder().wrap(new ByteArrayInputStream(bytes)));
|
||||
StreamInput in = new NamedWriteableAwareStreamInput(delegate, CURSOR_REGISTRY)) {
|
||||
Version version = Version.readVersion(in);
|
||||
if (version.after(Version.CURRENT)) {
|
||||
throw new RuntimeException("Unsupported scroll version " + version);
|
||||
}
|
||||
in.setVersion(version);
|
||||
return in.readNamedWriteable(Cursor.class);
|
||||
} catch (IOException ex) {
|
||||
throw new RuntimeException("unexpected failure deconding cursor", ex);
|
||||
}
|
||||
}
|
||||
}
|
|
@ -0,0 +1,99 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License;
|
||||
* you may not use this file except in compliance with the Elastic License.
|
||||
*/
|
||||
package org.elasticsearch.xpack.sql.session;
|
||||
|
||||
import org.elasticsearch.Version;
|
||||
import org.elasticsearch.common.io.stream.NamedWriteable;
|
||||
import org.elasticsearch.common.io.stream.NamedWriteableAwareStreamInput;
|
||||
import org.elasticsearch.common.io.stream.NamedWriteableRegistry;
|
||||
import org.elasticsearch.common.io.stream.OutputStreamStreamOutput;
|
||||
import org.elasticsearch.common.io.stream.StreamInput;
|
||||
import org.elasticsearch.common.io.stream.StreamOutput;
|
||||
import org.elasticsearch.xpack.sql.SqlIllegalArgumentException;
|
||||
import org.elasticsearch.xpack.sql.execution.search.CompositeAggregationCursor;
|
||||
import org.elasticsearch.xpack.sql.execution.search.ScrollCursor;
|
||||
import org.elasticsearch.xpack.sql.execution.search.extractor.BucketExtractors;
|
||||
import org.elasticsearch.xpack.sql.execution.search.extractor.HitExtractors;
|
||||
import org.elasticsearch.xpack.sql.expression.function.scalar.Processors;
|
||||
import org.elasticsearch.xpack.sql.plugin.CliFormatterCursor;
|
||||
|
||||
import java.io.ByteArrayOutputStream;
|
||||
import java.io.OutputStream;
|
||||
import java.nio.charset.StandardCharsets;
|
||||
import java.util.ArrayList;
|
||||
import java.util.Base64;
|
||||
import java.util.List;
|
||||
|
||||
/**
|
||||
* Registry and utilities around {@link Cursor}s.
|
||||
*/
|
||||
public final class Cursors {
|
||||
|
||||
private static final NamedWriteableRegistry WRITEABLE_REGISTRY = new NamedWriteableRegistry(getNamedWriteables());
|
||||
|
||||
private Cursors() {};
|
||||
|
||||
/**
|
||||
* The {@link NamedWriteable}s required to deserialize {@link Cursor}s.
|
||||
*/
|
||||
public static List<NamedWriteableRegistry.Entry> getNamedWriteables() {
|
||||
List<NamedWriteableRegistry.Entry> entries = new ArrayList<>();
|
||||
|
||||
// cursors
|
||||
entries.add(new NamedWriteableRegistry.Entry(Cursor.class, EmptyCursor.NAME, in -> Cursor.EMPTY));
|
||||
entries.add(new NamedWriteableRegistry.Entry(Cursor.class, ScrollCursor.NAME, ScrollCursor::new));
|
||||
entries.add(new NamedWriteableRegistry.Entry(Cursor.class, CompositeAggregationCursor.NAME, CompositeAggregationCursor::new));
|
||||
entries.add(new NamedWriteableRegistry.Entry(Cursor.class, CliFormatterCursor.NAME, CliFormatterCursor::new));
|
||||
|
||||
// plus all their dependencies
|
||||
entries.addAll(Processors.getNamedWriteables());
|
||||
entries.addAll(HitExtractors.getNamedWriteables());
|
||||
entries.addAll(BucketExtractors.getNamedWriteables());
|
||||
|
||||
return entries;
|
||||
}
|
||||
|
||||
/**
|
||||
* Write a {@linkplain Cursor} to a string for serialization across xcontent.
|
||||
*/
|
||||
public static String encodeToString(Version version, Cursor info) {
|
||||
if (info == Cursor.EMPTY) {
|
||||
return "";
|
||||
}
|
||||
try (ByteArrayOutputStream os = new ByteArrayOutputStream()) {
|
||||
try (OutputStream base64 = Base64.getEncoder().wrap(os); StreamOutput out = new OutputStreamStreamOutput(base64)) {
|
||||
Version.writeVersion(version, out);
|
||||
out.writeNamedWriteable(info);
|
||||
}
|
||||
return os.toString(StandardCharsets.UTF_8.name());
|
||||
} catch (Exception ex) {
|
||||
throw new SqlIllegalArgumentException("Unexpected failure retriving next page", ex);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Read a {@linkplain Cursor} from a string.
|
||||
*/
|
||||
public static Cursor decodeFromString(String info) {
|
||||
if (info.isEmpty()) {
|
||||
return Cursor.EMPTY;
|
||||
}
|
||||
byte[] bytes = info.getBytes(StandardCharsets.UTF_8);
|
||||
try (StreamInput in = new NamedWriteableAwareStreamInput(StreamInput.wrap(Base64.getDecoder().decode(bytes)), WRITEABLE_REGISTRY)) {
|
||||
Version version = Version.readVersion(in);
|
||||
if (version.after(Version.CURRENT)) {
|
||||
throw new SqlIllegalArgumentException("Unsupported cursor version " + version);
|
||||
}
|
||||
in.setVersion(version);
|
||||
return in.readNamedWriteable(Cursor.class);
|
||||
} catch (SqlIllegalArgumentException ex) {
|
||||
throw ex;
|
||||
} catch (Exception ex) {
|
||||
throw new SqlIllegalArgumentException("Unexpected failure decoding cursor", ex);
|
||||
}
|
||||
}
|
||||
}
|
|
@ -7,6 +7,7 @@ package org.elasticsearch.xpack.sql.session;
|
|||
|
||||
import org.elasticsearch.action.ActionListener;
|
||||
import org.elasticsearch.client.Client;
|
||||
import org.elasticsearch.common.io.stream.NamedWriteableRegistry;
|
||||
import org.elasticsearch.common.io.stream.StreamOutput;
|
||||
|
||||
import java.io.IOException;
|
||||
|
@ -30,7 +31,7 @@ class EmptyCursor implements Cursor {
|
|||
}
|
||||
|
||||
@Override
|
||||
public void nextPage(Configuration cfg, Client client, ActionListener<RowSet> listener) {
|
||||
public void nextPage(Configuration cfg, Client client, NamedWriteableRegistry registry, ActionListener<RowSet> listener) {
|
||||
throw new IllegalArgumentException("there is no next page");
|
||||
}
|
||||
|
||||
|
|
|
@ -5,14 +5,14 @@
|
|||
*/
|
||||
package org.elasticsearch.xpack.sql.session;
|
||||
|
||||
import java.util.ArrayList;
|
||||
import java.util.List;
|
||||
|
||||
import org.elasticsearch.xpack.sql.expression.Attribute;
|
||||
import org.elasticsearch.xpack.sql.type.DataType;
|
||||
import org.elasticsearch.xpack.sql.type.Schema;
|
||||
import org.elasticsearch.xpack.sql.util.Check;
|
||||
|
||||
import java.util.ArrayList;
|
||||
import java.util.List;
|
||||
|
||||
public abstract class Rows {
|
||||
|
||||
public static Schema schema(List<Attribute> attr) {
|
||||
|
@ -40,8 +40,12 @@ public abstract class Rows {
|
|||
}
|
||||
|
||||
public static SchemaRowSet singleton(List<Attribute> attrs, Object... values) {
|
||||
Check.isTrue(attrs.size() == values.length, "Schema %s and values %s are out of sync", attrs, values);
|
||||
return new SingletonRowSet(schema(attrs), values);
|
||||
return singleton(schema(attrs), values);
|
||||
}
|
||||
|
||||
public static SchemaRowSet singleton(Schema schema, Object... values) {
|
||||
Check.isTrue(schema.size() == values.length, "Schema {} and values {} are out of sync", schema, values);
|
||||
return new SingletonRowSet(schema, values);
|
||||
}
|
||||
|
||||
public static SchemaRowSet empty(Schema schema) {
|
||||
|
|
|
@ -17,7 +17,7 @@ public class SingletonExecutable implements Executable {
|
|||
private final Object[] values;
|
||||
|
||||
public SingletonExecutable(List<Attribute> output, Object... values) {
|
||||
Check.isTrue(output.size() == values.length, "Output %s and values %s are out of sync", output, values);
|
||||
Check.isTrue(output.size() == values.length, "Attributes {} and values {} are out of sync", output, values);
|
||||
this.output = output;
|
||||
this.values = values;
|
||||
}
|
||||
|
|
|
@ -5,6 +5,9 @@
|
|||
*/
|
||||
package org.elasticsearch.xpack.sql.type;
|
||||
|
||||
import org.elasticsearch.xpack.sql.type.Schema.Entry;
|
||||
import org.elasticsearch.xpack.sql.util.Check;
|
||||
|
||||
import java.util.Iterator;
|
||||
import java.util.List;
|
||||
import java.util.NoSuchElementException;
|
||||
|
@ -13,9 +16,6 @@ import java.util.Spliterators;
|
|||
import java.util.stream.Stream;
|
||||
import java.util.stream.StreamSupport;
|
||||
|
||||
import org.elasticsearch.xpack.sql.type.Schema.Entry;
|
||||
import org.elasticsearch.xpack.sql.util.Check;
|
||||
|
||||
import static java.util.Collections.emptyList;
|
||||
|
||||
public class Schema implements Iterable<Entry> {
|
||||
|
@ -51,7 +51,7 @@ public class Schema implements Iterable<Entry> {
|
|||
private final List<DataType> types;
|
||||
|
||||
public Schema(List<String> names, List<DataType> types) {
|
||||
Check.isTrue(names.size() == types.size(), "Different # of names %s vs types %s", names, types);
|
||||
Check.isTrue(names.size() == types.size(), "Different # of names {} vs types {}", names, types);
|
||||
this.types = types;
|
||||
this.names = names;
|
||||
}
|
||||
|
|
|
@ -0,0 +1,70 @@
|
|||
/*
|
||||
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
|
||||
* or more contributor license agreements. Licensed under the Elastic License;
|
||||
* you may not use this file except in compliance with the Elastic License.
|
||||
*/
|
||||
package org.elasticsearch.xpack.sql.execution.search;
|
||||
|
||||
import org.elasticsearch.Version;
|
||||
import org.elasticsearch.common.io.stream.NamedWriteableRegistry;
|
||||
import org.elasticsearch.common.io.stream.Writeable.Reader;
|
||||
import org.elasticsearch.test.AbstractWireSerializingTestCase;
|
||||
import org.elasticsearch.xpack.sql.execution.search.extractor.BucketExtractor;
|
||||
import org.elasticsearch.xpack.sql.execution.search.extractor.CompositeKeyExtractorTests;
|
||||
import org.elasticsearch.xpack.sql.execution.search.extractor.ConstantExtractorTests;
|
||||
import org.elasticsearch.xpack.sql.execution.search.extractor.MetricAggExtractorTests;
|
||||
import org.elasticsearch.xpack.sql.session.Cursors;
|
||||
|
||||
import java.io.IOException;
|
||||
import java.util.ArrayList;
|
||||
import java.util.List;
|
||||
import java.util.function.Supplier;
|
||||
|
||||
public class CompositeAggregationCursorTests extends AbstractWireSerializingTestCase<CompositeAggregationCursor> {
|
||||
public static CompositeAggregationCursor randomCompositeCursor() {
|
||||
int extractorsSize = between(1, 20);
|
||||
List<BucketExtractor> extractors = new ArrayList<>(extractorsSize);
|
||||
for (int i = 0; i < extractorsSize; i++) {
|
||||
extractors.add(randomBucketExtractor());
|
||||
}
|
||||
return new CompositeAggregationCursor(new byte[randomInt(256)], extractors, randomIntBetween(10, 1024), randomAlphaOfLength(5));
|
||||
}
|
||||
|
||||
static BucketExtractor randomBucketExtractor() {
|
||||
List<Supplier<BucketExtractor>> options = new ArrayList<>();
|
||||
options.add(ConstantExtractorTests::randomConstantExtractor);
|
||||
options.add(MetricAggExtractorTests::randomMetricAggExtractor);
|
||||
options.add(CompositeKeyExtractorTests::randomCompositeKeyExtractor);
|
||||
return randomFrom(options).get();
|
||||
}
|
||||
|
||||
@Override
|
||||
protected CompositeAggregationCursor mutateInstance(CompositeAggregationCursor instance) throws IOException {
|
||||
return new CompositeAggregationCursor(instance.next(), instance.extractors(), randomIntBetween(1, 512), instance.indices());
|
||||
}
|
||||
|
||||
@Override
|
||||
protected NamedWriteableRegistry getNamedWriteableRegistry() {
|
||||
return new NamedWriteableRegistry(Cursors.getNamedWriteables());
|
||||
}
|
||||
|
||||
@Override
|
||||
protected CompositeAggregationCursor createTestInstance() {
|
||||
return randomCompositeCursor();
|
||||
}
|
||||
|
||||
@Override
|
||||
protected Reader<CompositeAggregationCursor> instanceReader() {
|
||||
return CompositeAggregationCursor::new;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected CompositeAggregationCursor copyInstance(CompositeAggregationCursor instance, Version version) throws IOException {
|
||||
/* Randomly choose between internal protocol round trip and String based
|
||||
* round trips used to toXContent. */
|
||||
if (randomBoolean()) {
|
||||
return super.copyInstance(instance, version);
|
||||
}
|
||||
return (CompositeAggregationCursor) Cursors.decodeFromString(Cursors.encodeToString(version, instance));
|
||||
}
|
||||
}
|
|
@ -11,12 +11,14 @@ import org.elasticsearch.action.search.ClearScrollRequest;
|
|||
import org.elasticsearch.action.support.PlainActionFuture;
|
||||
import org.elasticsearch.client.Client;
|
||||
import org.elasticsearch.test.ESTestCase;
|
||||
import org.elasticsearch.xpack.sql.SqlException;
|
||||
import org.elasticsearch.xpack.sql.plugin.CliFormatter;
|
||||
import org.elasticsearch.xpack.sql.plugin.CliFormatterCursor;
|
||||
import org.elasticsearch.xpack.sql.plugin.ColumnInfo;
|
||||
import org.elasticsearch.xpack.sql.plugin.SqlQueryResponse;
|
||||
import org.elasticsearch.xpack.sql.session.Configuration;
|
||||
import org.elasticsearch.xpack.sql.session.Cursor;
|
||||
import org.elasticsearch.xpack.sql.session.Cursors;
|
||||
import org.mockito.ArgumentCaptor;
|
||||
|
||||
import java.sql.JDBCType;
|
||||
|
@ -90,16 +92,14 @@ public class CursorTests extends ESTestCase {
|
|||
|
||||
public void testVersionHandling() {
|
||||
Cursor cursor = randomNonEmptyCursor();
|
||||
assertEquals(cursor, Cursor.decodeFromString(Cursor.encodeToString(Version.CURRENT, cursor)));
|
||||
assertEquals(cursor, Cursors.decodeFromString(Cursors.encodeToString(Version.CURRENT, cursor)));
|
||||
|
||||
Version nextMinorVersion = Version.fromId(Version.CURRENT.id + 10000);
|
||||
|
||||
String encodedWithWrongVersion = Cursor.encodeToString(nextMinorVersion, cursor);
|
||||
RuntimeException exception = expectThrows(RuntimeException.class, () -> {
|
||||
Cursor.decodeFromString(encodedWithWrongVersion);
|
||||
});
|
||||
String encodedWithWrongVersion = Cursors.encodeToString(nextMinorVersion, cursor);
|
||||
SqlException exception = expectThrows(SqlException.class, () -> Cursors.decodeFromString(encodedWithWrongVersion));
|
||||
|
||||
assertEquals(exception.getMessage(), "Unsupported scroll version " + nextMinorVersion);
|
||||
assertEquals("Unsupported cursor version " + nextMinorVersion, exception.getMessage());
|
||||
}
|
||||
|
||||
|
||||
|
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue