From 24339eb640d9e177b3ffc6619822bed612d58932 Mon Sep 17 00:00:00 2001 From: David Roberts Date: Tue, 22 Feb 2022 16:59:09 +0000 Subject: [PATCH] [ML] Text structure finder caps exclude lines pattern at 1000 characters Because of the way Filebeat parses CSV files the text structure finder needs to generate a regular expression that will ignore the header row of the CSV file. It does this by concatenating the column names separated by the delimiter with optional quoting. However, if there are hundreds of columns this can lead to a very long regular expression, potentially one that cannot be evaluated by some programming languages. This change limits the length of the regular expression to 1000 characters by only including elements for the first few columns when there are many. Matching 1000 characters of header should be sufficient to reliably identify the header row even when it is much longer. It is extremely unlikely that there would be a data row where the first 1000 characters exactly matched the header but then subsequent fields diverged. Fixes #83434 --- .../DelimitedTextStructureFinder.java | 49 ++++++++++++++----- .../DelimitedTextStructureFinderTests.java | 23 +++++++++ 2 files changed, 60 insertions(+), 12 deletions(-) diff --git a/x-pack/plugin/text-structure/src/main/java/org/elasticsearch/xpack/textstructure/structurefinder/DelimitedTextStructureFinder.java b/x-pack/plugin/text-structure/src/main/java/org/elasticsearch/xpack/textstructure/structurefinder/DelimitedTextStructureFinder.java index 6d54fe73c95c..e1c425c81963 100644 --- a/x-pack/plugin/text-structure/src/main/java/org/elasticsearch/xpack/textstructure/structurefinder/DelimitedTextStructureFinder.java +++ b/x-pack/plugin/text-structure/src/main/java/org/elasticsearch/xpack/textstructure/structurefinder/DelimitedTextStructureFinder.java @@ -33,7 +33,8 @@ public class DelimitedTextStructureFinder implements TextStructureFinder { - private static final String REGEX_NEEDS_ESCAPE_PATTERN = "([\\\\|()\\[\\]{}^$.+*?])"; + static final int MAX_EXCLUDE_LINES_PATTERN_LENGTH = 1000; + static final String REGEX_NEEDS_ESCAPE_PATTERN = "([\\\\|()\\[\\]{}^$.+*?])"; private static final int MAX_LEVENSHTEIN_COMPARISONS = 100; private static final int LONG_FIELD_THRESHOLD = 100; private final List sampleMessages; @@ -137,20 +138,11 @@ static DelimitedTextStructureFinder makeDelimitedTextStructureFinder( .setColumnNames(columnNamesList); String quote = String.valueOf(quoteChar); - String twoQuotes = quote + quote; String quotePattern = quote.replaceAll(REGEX_NEEDS_ESCAPE_PATTERN, "\\\\$1"); String optQuotePattern = quotePattern + "?"; String delimiterPattern = (delimiter == '\t') ? "\\t" : String.valueOf(delimiter).replaceAll(REGEX_NEEDS_ESCAPE_PATTERN, "\\\\$1"); if (isHeaderInText) { - structureBuilder.setExcludeLinesPattern( - "^" - + Arrays.stream(header) - .map( - column -> optQuotePattern + column.replace(quote, twoQuotes).replaceAll(REGEX_NEEDS_ESCAPE_PATTERN, "\\\\$1") - + optQuotePattern - ) - .collect(Collectors.joining(delimiterPattern)) - ); + structureBuilder.setExcludeLinesPattern(makeExcludeLinesPattern(header, quote, optQuotePattern, delimiterPattern)); } if (trimFields) { @@ -413,7 +405,7 @@ private static boolean isFirstRowUnusual(List explanation, List maxLengthOfFields) { + excludeLinesPattern.append(".*"); + break; + } + excludeLinesPattern.append(delimiterPattern).append(columnPattern); + } + } + return excludeLinesPattern.toString(); + } } diff --git a/x-pack/plugin/text-structure/src/test/java/org/elasticsearch/xpack/textstructure/structurefinder/DelimitedTextStructureFinderTests.java b/x-pack/plugin/text-structure/src/test/java/org/elasticsearch/xpack/textstructure/structurefinder/DelimitedTextStructureFinderTests.java index 3530c94c8ede..9b94ed02515a 100644 --- a/x-pack/plugin/text-structure/src/test/java/org/elasticsearch/xpack/textstructure/structurefinder/DelimitedTextStructureFinderTests.java +++ b/x-pack/plugin/text-structure/src/test/java/org/elasticsearch/xpack/textstructure/structurefinder/DelimitedTextStructureFinderTests.java @@ -25,9 +25,12 @@ import static org.elasticsearch.xpack.textstructure.structurefinder.TimestampFormatFinder.stringToNumberPosBitSet; import static org.hamcrest.Matchers.arrayContaining; import static org.hamcrest.Matchers.contains; +import static org.hamcrest.Matchers.endsWith; import static org.hamcrest.Matchers.equalTo; import static org.hamcrest.Matchers.hasKey; +import static org.hamcrest.Matchers.lessThanOrEqualTo; import static org.hamcrest.Matchers.not; +import static org.hamcrest.Matchers.startsWith; public class DelimitedTextStructureFinderTests extends TextStructureTestCase { @@ -1122,6 +1125,26 @@ public void testMultilineStartPatternDeterminationTooHard() { assertThat(explanation, contains("Failed to create a suitable multi-line start pattern")); } + public void testMakeExcludeLinesPattern() { + + String[] header = generateRandomStringArray(1000, randomIntBetween(5, 50), false, false); + String quote = randomFrom("\"", "'"); + String quotePattern = quote.replaceAll(DelimitedTextStructureFinder.REGEX_NEEDS_ESCAPE_PATTERN, "\\\\$1"); + String optQuotePattern = quotePattern + "?"; + char delimiter = randomFrom(',', ';', '\t', '|'); + String delimiterPattern = (delimiter == '\t') + ? "\\t" + : String.valueOf(delimiter).replaceAll(DelimitedTextStructureFinder.REGEX_NEEDS_ESCAPE_PATTERN, "\\\\$1"); + + String excludeLinesPattern = DelimitedTextStructureFinder.makeExcludeLinesPattern(header, quote, optQuotePattern, delimiterPattern); + + assertThat(excludeLinesPattern, startsWith("^")); + assertThat(excludeLinesPattern.length(), lessThanOrEqualTo(DelimitedTextStructureFinder.MAX_EXCLUDE_LINES_PATTERN_LENGTH)); + if (excludeLinesPattern.contains(header[header.length - 1]) == false) { + assertThat(excludeLinesPattern, endsWith(".*")); + } + } + static Map randomCsvProcessorSettings() { String field = randomAlphaOfLength(10); return DelimitedTextStructureFinder.makeCsvProcessorSettings(