Skip to content

Commit

Permalink
ESQL: Limit size of query (elastic#117898)
Browse files Browse the repository at this point in the history
Queries bigger than a mb tend to take a lot of memory. In the worse case
it's an astounding amount of memory.
  • Loading branch information
nik9000 authored Dec 4, 2024
1 parent ec27e20 commit bbf986d
Show file tree
Hide file tree
Showing 4 changed files with 50 additions and 1 deletion.
5 changes: 5 additions & 0 deletions docs/changelog/117898.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
pr: 117898
summary: Limit size of query
area: ES|QL
type: bug
issues: []
Original file line number Diff line number Diff line change
Expand Up @@ -194,6 +194,13 @@ private void assertCircuitBreaks(ThrowingRunnable r) throws IOException {
);
}

private void assertParseFailure(ThrowingRunnable r) throws IOException {
ResponseException e = expectThrows(ResponseException.class, r);
Map<?, ?> map = responseAsMap(e.getResponse());
logger.info("expected parse failure {}", map);
assertMap(map, matchesMap().entry("status", 400).entry("error", matchesMap().extraOk().entry("type", "parsing_exception")));
}

private Response sortByManyLongs(int count) throws IOException {
logger.info("sorting by {} longs", count);
return query(makeSortByManyLongs(count).toString(), null);
Expand Down Expand Up @@ -318,6 +325,13 @@ public void testManyConcatFromRow() throws IOException {
assertManyStrings(resp, strings);
}

/**
* Fails to parse a huge huge query.
*/
public void testHugeHugeManyConcatFromRow() throws IOException {
assertParseFailure(() -> manyConcat("ROW a=9999, b=9999, c=9999, d=9999, e=9999", 50000));
}

/**
* Tests that generate many moderately long strings.
*/
Expand Down Expand Up @@ -378,6 +392,13 @@ public void testManyRepeatFromRow() throws IOException {
assertManyStrings(resp, strings);
}

/**
* Fails to parse a huge huge query.
*/
public void testHugeHugeManyRepeatFromRow() throws IOException {
assertParseFailure(() -> manyRepeat("ROW a = 99", 100000));
}

/**
* Tests that generate many moderately long strings.
*/
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -33,6 +33,15 @@ public class EsqlParser {

private static final Logger log = LogManager.getLogger(EsqlParser.class);

/**
* Maximum number of characters in an ESQL query. Antlr may parse the entire
* query into tokens to make the choices, buffering the world. There's a lot we
* can do in the grammar to prevent that, but let's be paranoid and assume we'll
* fail at preventing antlr from slurping in the world. Instead, let's make sure
* that the world just isn't that big.
*/
public static final int MAX_LENGTH = 1_000_000;

private EsqlConfig config = new EsqlConfig();

public EsqlConfig config() {
Expand Down Expand Up @@ -60,8 +69,14 @@ private <T> T invokeParser(
Function<EsqlBaseParser, ParserRuleContext> parseFunction,
BiFunction<AstBuilder, ParserRuleContext, T> result
) {
if (query.length() > MAX_LENGTH) {
throw new org.elasticsearch.xpack.esql.core.ParsingException(
"ESQL statement is too large [{} characters > {}]",
query.length(),
MAX_LENGTH
);
}
try {
// new CaseChangingCharStream()
EsqlBaseLexer lexer = new EsqlBaseLexer(CharStreams.fromString(query));

lexer.removeErrorListeners();
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -103,6 +103,14 @@ public void testInlineCast() throws IOException {
logger.info("Wrote to file: {}", file);
}

public void testTooBigQuery() {
StringBuilder query = new StringBuilder("FROM foo | EVAL a = a");
while (query.length() < EsqlParser.MAX_LENGTH) {
query.append(", a = CONCAT(a, a)");
}
assertEquals("-1:0: ESQL statement is too large [1000011 characters > 1000000]", error(query.toString()));
}

private String functionName(EsqlFunctionRegistry registry, Expression functionCall) {
for (FunctionDefinition def : registry.listFunctions()) {
if (functionCall.getClass().equals(def.clazz())) {
Expand Down

0 comments on commit bbf986d

Please sign in to comment.