Skip to content

Commit

Permalink
ESQL: Limit size of query
Browse files Browse the repository at this point in the history
Queries bigger than a mb tend to take a lot of memory.
  • Loading branch information
nik9000 committed Dec 3, 2024
1 parent cca7051 commit 88f8d5d
Show file tree
Hide file tree
Showing 2 changed files with 33 additions and 1 deletion.
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.
*/
private static final int MAX_LENGTH = 1_000_000;

private EsqlConfig config = new EsqlConfig();

public EsqlConfig config() {
Expand Down Expand Up @@ -60,8 +69,10 @@ private <T> T invokeParser(
Function<EsqlBaseParser, ParserRuleContext> parseFunction,
BiFunction<AstBuilder, ParserRuleContext, T> result
) {
if (query.length() > MAX_LENGTH) {
throw new 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

0 comments on commit 88f8d5d

Please sign in to comment.